api key logic works correctly
This commit is contained in:
parent
f78e816f20
commit
9b7adf9d8c
|
@ -55,11 +55,12 @@
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@budibase/bbui": "^1.17.0",
|
"@budibase/bbui": "^1.18.0",
|
||||||
"@budibase/client": "^0.1.1",
|
"@budibase/client": "^0.1.1",
|
||||||
"@budibase/colorpicker": "^1.0.1",
|
"@budibase/colorpicker": "^1.0.1",
|
||||||
"@nx-js/compiler-util": "^2.0.0",
|
"@nx-js/compiler-util": "^2.0.0",
|
||||||
"@sentry/browser": "5.19.1",
|
"@sentry/browser": "5.19.1",
|
||||||
|
"@svelteschool/svelte-forms": "^0.7.0",
|
||||||
"codemirror": "^5.51.0",
|
"codemirror": "^5.51.0",
|
||||||
"date-fns": "^1.29.0",
|
"date-fns": "^1.29.0",
|
||||||
"deepmerge": "^4.2.2",
|
"deepmerge": "^4.2.2",
|
||||||
|
@ -74,7 +75,8 @@
|
||||||
"string_decoder": "^1.2.0",
|
"string_decoder": "^1.2.0",
|
||||||
"svelte-portal": "^0.1.0",
|
"svelte-portal": "^0.1.0",
|
||||||
"svelte-simple-modal": "^0.4.2",
|
"svelte-simple-modal": "^0.4.2",
|
||||||
"uikit": "^3.1.7"
|
"uikit": "^3.1.7",
|
||||||
|
"yup": "^0.29.2"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@babel/core": "^7.5.5",
|
"@babel/core": "^7.5.5",
|
||||||
|
|
|
@ -1,4 +1,14 @@
|
||||||
|
<script context="module">
|
||||||
|
// Init store here so that it doesn't get destroyed and removed when the component instance is destroyed. This is to make sure no data is lost if the user closes the model
|
||||||
|
import { writable } from "svelte/store"
|
||||||
|
|
||||||
|
export const createAppStore = writable({ currentStep: 0, values: {} })
|
||||||
|
</script>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
|
import { string, object } from "yup"
|
||||||
|
import api from "builderStore/api"
|
||||||
|
import Form from "@svelteschool/svelte-forms"
|
||||||
import Spinner from "components/common/Spinner.svelte"
|
import Spinner from "components/common/Spinner.svelte"
|
||||||
import { API, Info, User } from "./Steps"
|
import { API, Info, User } from "./Steps"
|
||||||
import Indicator from "./Indicator.svelte"
|
import Indicator from "./Indicator.svelte"
|
||||||
|
@ -10,52 +20,136 @@
|
||||||
import { post } from "builderStore/api"
|
import { post } from "builderStore/api"
|
||||||
import analytics from "../../analytics"
|
import analytics from "../../analytics"
|
||||||
|
|
||||||
export let hasAPIKey = false
|
|
||||||
|
|
||||||
const { open, close } = getContext("simple-modal")
|
const { open, close } = getContext("simple-modal")
|
||||||
|
|
||||||
let name = ""
|
export let hasKey
|
||||||
let description = ""
|
|
||||||
let loading = false
|
|
||||||
let error = {}
|
|
||||||
|
|
||||||
const createNewApp = async () => {
|
let submitting = false
|
||||||
if ((name.length > 100 || name.length < 1) && description.length < 1) {
|
let errors = {}
|
||||||
error = {
|
let validationErrors = {}
|
||||||
name: true,
|
let validationSchemas = [
|
||||||
description: true,
|
{
|
||||||
|
apiKey: string().required("Please enter your API key."),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
applicationName: string().required("Your application must have a name."),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
username: string().required("Your application needs a first user."),
|
||||||
|
password: string().required(
|
||||||
|
"Please enter a password for your first user."
|
||||||
|
),
|
||||||
|
accessLevelId: string().required(
|
||||||
|
"You need to select an access level for your user."
|
||||||
|
),
|
||||||
|
},
|
||||||
|
]
|
||||||
|
|
||||||
|
let steps = [
|
||||||
|
{
|
||||||
|
component: API,
|
||||||
|
errors,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
component: Info,
|
||||||
|
errors,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
component: User,
|
||||||
|
errors,
|
||||||
|
},
|
||||||
|
]
|
||||||
|
|
||||||
|
if (hasKey) {
|
||||||
|
validationSchemas.shift()
|
||||||
|
validationSchemas = validationSchemas
|
||||||
|
steps.shift()
|
||||||
|
steps = steps
|
||||||
}
|
}
|
||||||
} else if (description.length < 1) {
|
|
||||||
error = {
|
// Handles form navigation
|
||||||
name: false,
|
const back = () => {
|
||||||
description: true,
|
if ($createAppStore.currentStep > 0) {
|
||||||
|
$createAppStore.currentStep -= 1
|
||||||
}
|
}
|
||||||
} else if (name.length > 100 || name.length < 1) {
|
|
||||||
error = {
|
|
||||||
name: true,
|
|
||||||
}
|
}
|
||||||
} else {
|
const next = () => {
|
||||||
error = {}
|
$createAppStore.currentStep += 1
|
||||||
const data = { name, description }
|
}
|
||||||
loading = true
|
|
||||||
|
// $: errors = validationSchemas.validate(values);
|
||||||
|
$: getErrors(
|
||||||
|
$createAppStore.values,
|
||||||
|
validationSchemas[$createAppStore.currentStep]
|
||||||
|
)
|
||||||
|
|
||||||
|
async function getErrors(values, schema) {
|
||||||
try {
|
try {
|
||||||
const response = await post("/api/applications", data)
|
validationErrors = {}
|
||||||
|
await object(schema).validate(values, { abortEarly: false })
|
||||||
|
} catch (error) {
|
||||||
|
validationErrors = extractErrors(error)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const checkValidity = async (values, currentStep) => {
|
||||||
|
const validity = await object()
|
||||||
|
.shape(validationSchemas[currentStep])
|
||||||
|
.isValid(values)
|
||||||
|
currentStepIsValid = validity
|
||||||
|
|
||||||
|
// Check full form on last step
|
||||||
|
if (currentStep === steps.length - 1) {
|
||||||
|
// Make one big schema from all the small ones
|
||||||
|
const fullSchema = Object.assign({}, ...validationSchemas)
|
||||||
|
|
||||||
|
// Check full form schema
|
||||||
|
const formIsValid = await object()
|
||||||
|
.shape(fullSchema)
|
||||||
|
.isValid(values)
|
||||||
|
fullFormIsValid = formIsValid
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function signUp() {
|
||||||
|
submitting = true
|
||||||
|
try {
|
||||||
|
if (!hasKey) {
|
||||||
|
await updateKey(["budibase", $createAppStore.values.apiKey])
|
||||||
|
}
|
||||||
|
const response = await post("/api/applications", {
|
||||||
|
name: $createAppStore.values.applicationName,
|
||||||
|
})
|
||||||
const res = await response.json()
|
const res = await response.json()
|
||||||
|
|
||||||
analytics.captureEvent("web_app_created", {
|
analytics.captureEvent("web_app_created", {
|
||||||
name,
|
name,
|
||||||
description,
|
description,
|
||||||
appId: res._id,
|
appId: res._id,
|
||||||
})
|
})
|
||||||
$goto(`./${res._id}`)
|
console.log("App created!")
|
||||||
|
// $goto(`./${res._id}`)
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error(error)
|
console.error(error)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function updateKey([key, value]) {
|
||||||
|
console.log("Saving API Key")
|
||||||
|
const response = await api.put(`/api/keys/${key}`, { value })
|
||||||
|
const res = await response.json()
|
||||||
|
return res
|
||||||
}
|
}
|
||||||
|
|
||||||
let value
|
function extractErrors({ inner }) {
|
||||||
|
return inner.reduce((acc, err) => {
|
||||||
|
return { ...acc, [err.path]: err.message }
|
||||||
|
}, {})
|
||||||
|
}
|
||||||
|
|
||||||
|
let currentStepIsValid = false
|
||||||
|
let fullFormIsValid = false
|
||||||
|
$: checkValidity($createAppStore.values, $createAppStore.currentStep)
|
||||||
|
|
||||||
let onChange = () => {}
|
let onChange = () => {}
|
||||||
|
|
||||||
function _onCancel() {
|
function _onCancel() {
|
||||||
|
@ -65,21 +159,14 @@
|
||||||
async function _onOkay() {
|
async function _onOkay() {
|
||||||
await createNewApp()
|
await createNewApp()
|
||||||
}
|
}
|
||||||
|
|
||||||
let currentStep = 0
|
|
||||||
let steps = [
|
|
||||||
{ title: "Setup your API Key" },
|
|
||||||
{ title: "Create your first web app" },
|
|
||||||
{ title: "Create new user" },
|
|
||||||
]
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="container">
|
<div class="container">
|
||||||
<div class="sidebar">
|
<div class="sidebar">
|
||||||
{#each steps as { active, done }, i}
|
{#each steps as { active, done }, i}
|
||||||
<Indicator
|
<Indicator
|
||||||
active={currentStep === i}
|
active={$createAppStore.currentStep === i}
|
||||||
done={i < currentStep}
|
done={i < $createAppStore.currentStep}
|
||||||
step={i + 1} />
|
step={i + 1} />
|
||||||
{/each}
|
{/each}
|
||||||
</div>
|
</div>
|
||||||
|
@ -88,38 +175,41 @@
|
||||||
<h3 class="header">Get Started with Budibase</h3>
|
<h3 class="header">Get Started with Budibase</h3>
|
||||||
</div>
|
</div>
|
||||||
<div class="step">
|
<div class="step">
|
||||||
<Input
|
<Form bind:values={$createAppStore.values}>
|
||||||
error={error.name}
|
{#each steps as step, i (i)}
|
||||||
name="name"
|
<div class:hidden={$createAppStore.currentStep !== i}>
|
||||||
label="Name"
|
<svelte:component
|
||||||
placeholder="Enter application name"
|
this={step.component}
|
||||||
on:change={e => (name = e.target.value)}
|
{validationErrors}
|
||||||
on:input={e => (name = e.target.value)} />
|
options={step.options}
|
||||||
<TextArea
|
name={step.name} />
|
||||||
bind:value={description}
|
</div>
|
||||||
name="description"
|
{/each}
|
||||||
label="Description"
|
</Form>
|
||||||
placeholder="Describe your application" />
|
|
||||||
{#if error.description}
|
|
||||||
<span class="error">
|
|
||||||
Please enter a short description of your application
|
|
||||||
</span>
|
|
||||||
{/if}
|
|
||||||
</div>
|
</div>
|
||||||
<div class="footer">
|
<div class="footer">
|
||||||
{#if currentStep !== 0}
|
{#if $createAppStore.currentStep > 0}
|
||||||
<Button primary thin on:click={_onOkay}>Back</Button>
|
<Button secondary on:click={back}>Back</Button>
|
||||||
{/if}
|
{/if}
|
||||||
<Button primary thin on:click={_onOkay}>Next</Button>
|
{#if $createAppStore.currentStep < steps.length - 1}
|
||||||
{#if currentStep + 1 === steps.length}
|
<Button secondary on:click={next} disabled={!currentStepIsValid}>
|
||||||
<Button primary thin on:click={_onOkay}>Launch</Button>
|
Next
|
||||||
|
</Button>
|
||||||
|
{/if}
|
||||||
|
{#if $createAppStore.currentStep === steps.length - 1}
|
||||||
|
<Button
|
||||||
|
secondary
|
||||||
|
on:click={signUp}
|
||||||
|
disabled={!fullFormIsValid || submitting}>
|
||||||
|
{submitting ? 'Loading...' : 'Submit'}
|
||||||
|
</Button>
|
||||||
{/if}
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="close-button" on:click={_onCancel}>
|
<div class="close-button" on:click={_onCancel}>
|
||||||
<CloseIcon />
|
<CloseIcon />
|
||||||
</div>
|
</div>
|
||||||
{#if loading}
|
{#if submitting}
|
||||||
<div in:fade class="spinner-container">
|
<div in:fade class="spinner-container">
|
||||||
<Spinner />
|
<Spinner />
|
||||||
<span class="spinner-text">Creating your app...</span>
|
<span class="spinner-text">Creating your app...</span>
|
||||||
|
@ -129,6 +219,7 @@
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
.container {
|
.container {
|
||||||
|
min-height: 600px;
|
||||||
display: grid;
|
display: grid;
|
||||||
grid-template-columns: 80px 1fr;
|
grid-template-columns: 80px 1fr;
|
||||||
position: relative;
|
position: relative;
|
||||||
|
@ -165,6 +256,7 @@
|
||||||
.body {
|
.body {
|
||||||
padding: 40px 60px 60px 60px;
|
padding: 40px 60px 60px 60px;
|
||||||
display: grid;
|
display: grid;
|
||||||
|
align-items: center;
|
||||||
grid-template-rows: auto 1fr auto;
|
grid-template-rows: auto 1fr auto;
|
||||||
}
|
}
|
||||||
.footer {
|
.footer {
|
||||||
|
@ -189,9 +281,8 @@
|
||||||
.spinner-text {
|
.spinner-text {
|
||||||
font-size: 2em;
|
font-size: 2em;
|
||||||
}
|
}
|
||||||
.error {
|
|
||||||
color: var(--red);
|
.hidden {
|
||||||
font-weight: bold;
|
display: none;
|
||||||
font-size: 0.8em;
|
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -0,0 +1,25 @@
|
||||||
|
<script>
|
||||||
|
import { Input } from "@budibase/bbui"
|
||||||
|
export let validationErrors
|
||||||
|
|
||||||
|
let blurred = { api: false }
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<h2>Setup your API Key</h2>
|
||||||
|
<div class="container">
|
||||||
|
<Input
|
||||||
|
on:input={() => (blurred.api = true)}
|
||||||
|
label="API Key"
|
||||||
|
name="apiKey"
|
||||||
|
placeholder="Enter your API Key"
|
||||||
|
type="password"
|
||||||
|
error={blurred.api && validationErrors.apiKey} />
|
||||||
|
<a target="_blank" href="https://portal.budi.live/">Get API Key</a>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.container {
|
||||||
|
display: grid;
|
||||||
|
grid-gap: 40px;
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -0,0 +1,24 @@
|
||||||
|
<script>
|
||||||
|
import { Input } from "@budibase/bbui"
|
||||||
|
export let validationErrors
|
||||||
|
|
||||||
|
let blurred = { appName: false }
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<h2>Create your first web app</h2>
|
||||||
|
<div class="container">
|
||||||
|
<Input
|
||||||
|
on:input={() => (blurred.appName = true)}
|
||||||
|
label="Web app name"
|
||||||
|
name="applicationName"
|
||||||
|
placeholder="Enter name of your web application"
|
||||||
|
type="name"
|
||||||
|
error={blurred.appName && validationErrors.applicationName} />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.container {
|
||||||
|
display: grid;
|
||||||
|
grid-gap: 40px;
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -0,0 +1,35 @@
|
||||||
|
<script>
|
||||||
|
import { Input, Select } from "@budibase/bbui"
|
||||||
|
export let validationErrors
|
||||||
|
|
||||||
|
let blurred = { username: false, password: false }
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<h2>Create new user</h2>
|
||||||
|
<div class="container">
|
||||||
|
<Input
|
||||||
|
on:input={() => (blurred.username = true)}
|
||||||
|
label="Username"
|
||||||
|
name="username"
|
||||||
|
placeholder="Username"
|
||||||
|
type="name"
|
||||||
|
error={blurred.username && validationErrors.username} />
|
||||||
|
<Input
|
||||||
|
on:input={() => (blurred.password = true)}
|
||||||
|
label="Password"
|
||||||
|
name="password"
|
||||||
|
placeholder="Password"
|
||||||
|
type="pasword"
|
||||||
|
error={blurred.password && validationErrors.password} />
|
||||||
|
<Select name="accessLevelId">
|
||||||
|
<option value="ADMIN">Admin</option>
|
||||||
|
<option value="POWER_USER">Power User</option>
|
||||||
|
</Select>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.container {
|
||||||
|
display: grid;
|
||||||
|
grid-gap: 40px;
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -1,6 +1,7 @@
|
||||||
<script>
|
<script>
|
||||||
import { getContext } from "svelte"
|
import { getContext } from "svelte"
|
||||||
import { store } from "builderStore"
|
import { store } from "builderStore"
|
||||||
|
import api from "builderStore/api"
|
||||||
import AppList from "components/start/AppList.svelte"
|
import AppList from "components/start/AppList.svelte"
|
||||||
import { onMount } from "svelte"
|
import { onMount } from "svelte"
|
||||||
import ActionButton from "components/common/ActionButton.svelte"
|
import ActionButton from "components/common/ActionButton.svelte"
|
||||||
|
@ -23,6 +24,24 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let hasKey
|
||||||
|
|
||||||
|
async function fetchKeys() {
|
||||||
|
const response = await api.get(`/api/keys/`)
|
||||||
|
const res = await response.json()
|
||||||
|
return res.budibase
|
||||||
|
}
|
||||||
|
|
||||||
|
async function checkIfKeysAndApps() {
|
||||||
|
const key = await fetchKeys()
|
||||||
|
const apps = await getApps()
|
||||||
|
if (key) {
|
||||||
|
hasKey = true
|
||||||
|
} else {
|
||||||
|
showCreateAppModal()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Handle create app modal
|
// Handle create app modal
|
||||||
const { open } = getContext("simple-modal")
|
const { open } = getContext("simple-modal")
|
||||||
|
|
||||||
|
@ -30,8 +49,7 @@
|
||||||
open(
|
open(
|
||||||
CreateAppModal,
|
CreateAppModal,
|
||||||
{
|
{
|
||||||
message: "What is your name?",
|
hasKey,
|
||||||
hasForm: true,
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
closeButton: false,
|
closeButton: false,
|
||||||
|
@ -42,6 +60,8 @@
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
checkIfKeysAndApps()
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="header">
|
<div class="header">
|
||||||
|
|
Loading…
Reference in New Issue