Merge branch 'master' of github.com:Budibase/budibase into BUDI-7656/add-migration
This commit is contained in:
commit
e3b1655ad0
|
@ -9,7 +9,7 @@ on:
|
||||||
jobs:
|
jobs:
|
||||||
ensure-is-master-tag:
|
ensure-is-master-tag:
|
||||||
name: Ensure is a master tag
|
name: Ensure is a master tag
|
||||||
runs-on: qa-arc-runner-set
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout monorepo
|
- name: Checkout monorepo
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
|
|
|
@ -17,6 +17,6 @@ version: 0.0.0
|
||||||
appVersion: 0.0.0
|
appVersion: 0.0.0
|
||||||
dependencies:
|
dependencies:
|
||||||
- name: couchdb
|
- name: couchdb
|
||||||
version: 4.3.0
|
version: 4.5.3
|
||||||
repository: https://apache.github.io/couchdb-helm
|
repository: https://apache.github.io/couchdb-helm
|
||||||
condition: services.couchdb.enabled
|
condition: services.couchdb.enabled
|
||||||
|
|
|
@ -74,6 +74,7 @@ http {
|
||||||
add_header X-Content-Type-Options nosniff always;
|
add_header X-Content-Type-Options nosniff always;
|
||||||
add_header X-XSS-Protection "1; mode=block" always;
|
add_header X-XSS-Protection "1; mode=block" always;
|
||||||
add_header Content-Security-Policy "${csp_default}; ${csp_script}; ${csp_style}; ${csp_object}; ${csp_base_uri}; ${csp_connect}; ${csp_font}; ${csp_frame}; ${csp_img}; ${csp_manifest}; ${csp_media}; ${csp_worker};" always;
|
add_header Content-Security-Policy "${csp_default}; ${csp_script}; ${csp_style}; ${csp_object}; ${csp_base_uri}; ${csp_connect}; ${csp_font}; ${csp_frame}; ${csp_img}; ${csp_manifest}; ${csp_media}; ${csp_worker};" always;
|
||||||
|
add_header Strict-Transport-Security "max-age=31536000; includeSubDomains; preload" always;
|
||||||
|
|
||||||
# upstreams
|
# upstreams
|
||||||
set $apps ${APPS_UPSTREAM_URL};
|
set $apps ${APPS_UPSTREAM_URL};
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
{
|
{
|
||||||
"version": "2.27.4",
|
"version": "2.27.6",
|
||||||
"npmClient": "yarn",
|
"npmClient": "yarn",
|
||||||
"packages": [
|
"packages": [
|
||||||
"packages/*",
|
"packages/*",
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
Subproject commit 39acfff42a063e5a8a7d58d36721ec3103e16348
|
Subproject commit a03225549e3ce61f43d0da878da162e08941b939
|
|
@ -14,6 +14,7 @@ import { v4 } from "uuid"
|
||||||
import { APP_PREFIX, APP_DEV_PREFIX } from "../db"
|
import { APP_PREFIX, APP_DEV_PREFIX } from "../db"
|
||||||
import fsp from "fs/promises"
|
import fsp from "fs/promises"
|
||||||
import { HeadObjectOutput } from "aws-sdk/clients/s3"
|
import { HeadObjectOutput } from "aws-sdk/clients/s3"
|
||||||
|
import { ReadableStream } from "stream/web"
|
||||||
|
|
||||||
const streamPipeline = promisify(stream.pipeline)
|
const streamPipeline = promisify(stream.pipeline)
|
||||||
// use this as a temporary store of buckets that are being created
|
// use this as a temporary store of buckets that are being created
|
||||||
|
@ -41,10 +42,7 @@ type UploadParams = BaseUploadParams & {
|
||||||
path?: string | PathLike
|
path?: string | PathLike
|
||||||
}
|
}
|
||||||
|
|
||||||
export type StreamTypes =
|
export type StreamTypes = ReadStream | NodeJS.ReadableStream
|
||||||
| ReadStream
|
|
||||||
| NodeJS.ReadableStream
|
|
||||||
| ReadableStream<Uint8Array>
|
|
||||||
|
|
||||||
export type StreamUploadParams = BaseUploadParams & {
|
export type StreamUploadParams = BaseUploadParams & {
|
||||||
stream?: StreamTypes
|
stream?: StreamTypes
|
||||||
|
@ -222,6 +220,9 @@ export async function streamUpload({
|
||||||
extra,
|
extra,
|
||||||
ttl,
|
ttl,
|
||||||
}: StreamUploadParams) {
|
}: StreamUploadParams) {
|
||||||
|
if (!stream) {
|
||||||
|
throw new Error("Stream to upload is invalid/undefined")
|
||||||
|
}
|
||||||
const extension = filename.split(".").pop()
|
const extension = filename.split(".").pop()
|
||||||
const objectStore = ObjectStore(bucketName)
|
const objectStore = ObjectStore(bucketName)
|
||||||
const bucketCreated = await createBucketIfNotExists(objectStore, bucketName)
|
const bucketCreated = await createBucketIfNotExists(objectStore, bucketName)
|
||||||
|
@ -251,14 +252,27 @@ export async function streamUpload({
|
||||||
: CONTENT_TYPE_MAP.txt
|
: CONTENT_TYPE_MAP.txt
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const bucket = sanitizeBucket(bucketName),
|
||||||
|
objKey = sanitizeKey(filename)
|
||||||
const params = {
|
const params = {
|
||||||
Bucket: sanitizeBucket(bucketName),
|
Bucket: bucket,
|
||||||
Key: sanitizeKey(filename),
|
Key: objKey,
|
||||||
Body: stream,
|
Body: stream,
|
||||||
ContentType: contentType,
|
ContentType: contentType,
|
||||||
...extra,
|
...extra,
|
||||||
}
|
}
|
||||||
return objectStore.upload(params).promise()
|
|
||||||
|
const details = await objectStore.upload(params).promise()
|
||||||
|
const headDetails = await objectStore
|
||||||
|
.headObject({
|
||||||
|
Bucket: bucket,
|
||||||
|
Key: objKey,
|
||||||
|
})
|
||||||
|
.promise()
|
||||||
|
return {
|
||||||
|
...details,
|
||||||
|
ContentLength: headDetails.ContentLength,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -57,6 +57,7 @@
|
||||||
class:fullWidth
|
class:fullWidth
|
||||||
class="spectrum-ActionButton spectrum-ActionButton--size{size}"
|
class="spectrum-ActionButton spectrum-ActionButton--size{size}"
|
||||||
class:active
|
class:active
|
||||||
|
class:disabled
|
||||||
{disabled}
|
{disabled}
|
||||||
on:longPress
|
on:longPress
|
||||||
on:click|preventDefault
|
on:click|preventDefault
|
||||||
|
@ -109,19 +110,22 @@
|
||||||
background: var(--spectrum-global-color-gray-300);
|
background: var(--spectrum-global-color-gray-300);
|
||||||
border-color: var(--spectrum-global-color-gray-500);
|
border-color: var(--spectrum-global-color-gray-500);
|
||||||
}
|
}
|
||||||
.noPadding {
|
|
||||||
padding: 0;
|
|
||||||
min-width: 0;
|
|
||||||
}
|
|
||||||
.spectrum-ActionButton--quiet {
|
.spectrum-ActionButton--quiet {
|
||||||
padding: 0 8px;
|
padding: 0 8px;
|
||||||
}
|
}
|
||||||
.spectrum-ActionButton--quiet.is-selected {
|
.spectrum-ActionButton--quiet.is-selected {
|
||||||
color: var(--spectrum-global-color-gray-900);
|
color: var(--spectrum-global-color-gray-900);
|
||||||
}
|
}
|
||||||
|
.noPadding {
|
||||||
|
padding: 0;
|
||||||
|
min-width: 0;
|
||||||
|
}
|
||||||
.is-selected:not(.emphasized) .spectrum-Icon {
|
.is-selected:not(.emphasized) .spectrum-Icon {
|
||||||
color: var(--spectrum-global-color-gray-900);
|
color: var(--spectrum-global-color-gray-900);
|
||||||
}
|
}
|
||||||
|
.is-selected.disabled .spectrum-Icon {
|
||||||
|
color: var(--spectrum-global-color-gray-500);
|
||||||
|
}
|
||||||
.tooltip {
|
.tooltip {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
pointer-events: none;
|
pointer-events: none;
|
||||||
|
|
|
@ -11,6 +11,7 @@
|
||||||
import {
|
import {
|
||||||
decodeJSBinding,
|
decodeJSBinding,
|
||||||
encodeJSBinding,
|
encodeJSBinding,
|
||||||
|
processObjectSync,
|
||||||
processStringSync,
|
processStringSync,
|
||||||
} from "@budibase/string-templates"
|
} from "@budibase/string-templates"
|
||||||
import { readableToRuntimeBinding } from "dataBinding"
|
import { readableToRuntimeBinding } from "dataBinding"
|
||||||
|
@ -153,13 +154,6 @@
|
||||||
debouncedEval(expression, context, snippets)
|
debouncedEval(expression, context, snippets)
|
||||||
}
|
}
|
||||||
|
|
||||||
const getBindingValue = (binding, context, snippets) => {
|
|
||||||
const js = `return $("${binding.runtimeBinding}")`
|
|
||||||
const hbs = encodeJSBinding(js)
|
|
||||||
const res = processStringSync(hbs, { ...context, snippets })
|
|
||||||
return JSON.stringify(res, null, 2)
|
|
||||||
}
|
|
||||||
|
|
||||||
const highlightJSON = json => {
|
const highlightJSON = json => {
|
||||||
return formatHighlight(json, {
|
return formatHighlight(json, {
|
||||||
keyColor: "#e06c75",
|
keyColor: "#e06c75",
|
||||||
|
@ -172,11 +166,27 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
const enrichBindings = (bindings, context, snippets) => {
|
const enrichBindings = (bindings, context, snippets) => {
|
||||||
return bindings.map(binding => {
|
// Create a single big array to enrich in one go
|
||||||
|
const bindingStrings = bindings.map(binding => {
|
||||||
|
if (binding.runtimeBinding.startsWith('trim "')) {
|
||||||
|
// Account for nasty hardcoded HBS bindings for roles, for legacy
|
||||||
|
// compatibility
|
||||||
|
return `{{ ${binding.runtimeBinding} }}`
|
||||||
|
} else {
|
||||||
|
return `{{ literal ${binding.runtimeBinding} }}`
|
||||||
|
}
|
||||||
|
})
|
||||||
|
const bindingEvauations = processObjectSync(bindingStrings, {
|
||||||
|
...context,
|
||||||
|
snippets,
|
||||||
|
})
|
||||||
|
|
||||||
|
// Enrich bindings with evaluations and highlighted HTML
|
||||||
|
return bindings.map((binding, idx) => {
|
||||||
if (!context) {
|
if (!context) {
|
||||||
return binding
|
return binding
|
||||||
}
|
}
|
||||||
const value = getBindingValue(binding, context, snippets)
|
const value = JSON.stringify(bindingEvauations[idx], null, 2)
|
||||||
return {
|
return {
|
||||||
...binding,
|
...binding,
|
||||||
value,
|
value,
|
||||||
|
|
|
@ -75,13 +75,6 @@
|
||||||
if (!context || !binding.value || binding.value === "") {
|
if (!context || !binding.value || binding.value === "") {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Roles have always been broken for JS. We need to exclude them from
|
|
||||||
// showing a popover as it will show "Error while executing JS".
|
|
||||||
if (binding.category === "Role") {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
stopHidingPopover()
|
stopHidingPopover()
|
||||||
popoverAnchor = target
|
popoverAnchor = target
|
||||||
hoverTarget = {
|
hoverTarget = {
|
||||||
|
|
|
@ -60,6 +60,7 @@
|
||||||
userLimitReachedModal
|
userLimitReachedModal
|
||||||
let searchEmail = undefined
|
let searchEmail = undefined
|
||||||
let selectedRows = []
|
let selectedRows = []
|
||||||
|
let selectedInvites = []
|
||||||
let bulkSaveResponse
|
let bulkSaveResponse
|
||||||
let customRenderers = [
|
let customRenderers = [
|
||||||
{ column: "email", component: EmailTableRenderer },
|
{ column: "email", component: EmailTableRenderer },
|
||||||
|
@ -123,7 +124,7 @@
|
||||||
return {}
|
return {}
|
||||||
}
|
}
|
||||||
let pendingSchema = JSON.parse(JSON.stringify(tblSchema))
|
let pendingSchema = JSON.parse(JSON.stringify(tblSchema))
|
||||||
pendingSchema.email.displayName = "Pending Invites"
|
pendingSchema.email.displayName = "Pending Users"
|
||||||
return pendingSchema
|
return pendingSchema
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -132,6 +133,7 @@
|
||||||
const { admin, builder, userGroups, apps } = invite.info
|
const { admin, builder, userGroups, apps } = invite.info
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
_id: invite.code,
|
||||||
email: invite.email,
|
email: invite.email,
|
||||||
builder,
|
builder,
|
||||||
admin,
|
admin,
|
||||||
|
@ -260,9 +262,26 @@
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
await users.bulkDelete(ids)
|
if (ids.length > 0) {
|
||||||
notifications.success(`Successfully deleted ${selectedRows.length} rows`)
|
await users.bulkDelete(ids)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (selectedInvites.length > 0) {
|
||||||
|
await users.removeInvites(
|
||||||
|
selectedInvites.map(invite => ({
|
||||||
|
code: invite._id,
|
||||||
|
}))
|
||||||
|
)
|
||||||
|
pendingInvites = await users.getInvites()
|
||||||
|
}
|
||||||
|
|
||||||
|
notifications.success(
|
||||||
|
`Successfully deleted ${
|
||||||
|
selectedRows.length + selectedInvites.length
|
||||||
|
} users`
|
||||||
|
)
|
||||||
selectedRows = []
|
selectedRows = []
|
||||||
|
selectedInvites = []
|
||||||
await fetch.refresh()
|
await fetch.refresh()
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
notifications.error("Error deleting users")
|
notifications.error("Error deleting users")
|
||||||
|
@ -328,15 +347,15 @@
|
||||||
</div>
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
<div class="controls-right">
|
<div class="controls-right">
|
||||||
<Search bind:value={searchEmail} placeholder="Search" />
|
{#if selectedRows.length > 0 || selectedInvites.length > 0}
|
||||||
{#if selectedRows.length > 0}
|
|
||||||
<DeleteRowsButton
|
<DeleteRowsButton
|
||||||
item="user"
|
item="user"
|
||||||
on:updaterows
|
on:updaterows
|
||||||
{selectedRows}
|
selectedRows={[...selectedRows, ...selectedInvites]}
|
||||||
deleteRows={deleteUsers}
|
deleteRows={deleteUsers}
|
||||||
/>
|
/>
|
||||||
{/if}
|
{/if}
|
||||||
|
<Search bind:value={searchEmail} placeholder="Search" />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<Table
|
<Table
|
||||||
|
@ -362,10 +381,12 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Table
|
<Table
|
||||||
|
bind:selectedRows={selectedInvites}
|
||||||
schema={pendingSchema}
|
schema={pendingSchema}
|
||||||
data={parsedInvites}
|
data={parsedInvites}
|
||||||
allowEditColumns={false}
|
allowEditColumns={false}
|
||||||
allowEditRows={false}
|
allowEditRows={false}
|
||||||
|
allowSelectRows={!readonly}
|
||||||
{customRenderers}
|
{customRenderers}
|
||||||
loading={!invitesLoaded}
|
loading={!invitesLoaded}
|
||||||
allowClickRows={false}
|
allowClickRows={false}
|
||||||
|
|
|
@ -38,6 +38,10 @@ export function createUsersStore() {
|
||||||
return API.inviteUsers(payload)
|
return API.inviteUsers(payload)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function removeInvites(payload) {
|
||||||
|
return API.removeUserInvites(payload)
|
||||||
|
}
|
||||||
|
|
||||||
async function acceptInvite(inviteCode, password, firstName, lastName) {
|
async function acceptInvite(inviteCode, password, firstName, lastName) {
|
||||||
return API.acceptInvite({
|
return API.acceptInvite({
|
||||||
inviteCode,
|
inviteCode,
|
||||||
|
@ -154,6 +158,7 @@ export function createUsersStore() {
|
||||||
onboard,
|
onboard,
|
||||||
fetchInvite,
|
fetchInvite,
|
||||||
getInvites,
|
getInvites,
|
||||||
|
removeInvites,
|
||||||
updateInvite,
|
updateInvite,
|
||||||
getUserCountByApp,
|
getUserCountByApp,
|
||||||
addAppBuilder,
|
addAppBuilder,
|
||||||
|
|
|
@ -127,7 +127,7 @@
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
}
|
}
|
||||||
label {
|
label {
|
||||||
white-space: nowrap;
|
word-wrap: break-word;
|
||||||
}
|
}
|
||||||
label.hidden {
|
label.hidden {
|
||||||
padding: 0;
|
padding: 0;
|
||||||
|
|
|
@ -234,6 +234,16 @@ export const buildUserEndpoints = API => ({
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Removes multiple user invites from Redis cache
|
||||||
|
*/
|
||||||
|
removeUserInvites: async inviteCodes => {
|
||||||
|
return await API.post({
|
||||||
|
url: "/api/global/users/multi/invite/delete",
|
||||||
|
body: inviteCodes,
|
||||||
|
})
|
||||||
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Accepts an invite to join the platform and creates a user.
|
* Accepts an invite to join the platform and creates a user.
|
||||||
* @param inviteCode the invite code sent in the email
|
* @param inviteCode the invite code sent in the email
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
<script>
|
<script>
|
||||||
import { getContext } from "svelte"
|
import { getContext } from "svelte"
|
||||||
import { ActionButton, Popover, Toggle, Icon } from "@budibase/bbui"
|
import { ActionButton, Popover, Icon } from "@budibase/bbui"
|
||||||
import { getColumnIcon } from "../lib/utils"
|
import { getColumnIcon } from "../lib/utils"
|
||||||
|
import ToggleActionButtonGroup from "./ToggleActionButtonGroup.svelte"
|
||||||
|
|
||||||
const { columns, datasource, stickyColumn, dispatch } = getContext("grid")
|
const { columns, datasource, stickyColumn, dispatch } = getContext("grid")
|
||||||
|
|
||||||
|
@ -11,31 +12,45 @@
|
||||||
$: anyHidden = $columns.some(col => !col.visible)
|
$: anyHidden = $columns.some(col => !col.visible)
|
||||||
$: text = getText($columns)
|
$: text = getText($columns)
|
||||||
|
|
||||||
const toggleColumn = async (column, visible) => {
|
const toggleColumn = async (column, permission) => {
|
||||||
datasource.actions.addSchemaMutation(column.name, { visible })
|
const visible = permission !== PERMISSION_OPTIONS.HIDDEN
|
||||||
await datasource.actions.saveSchemaMutations()
|
|
||||||
dispatch(visible ? "show-column" : "hide-column")
|
|
||||||
}
|
|
||||||
|
|
||||||
const toggleAll = async visible => {
|
datasource.actions.addSchemaMutation(column.name, { visible })
|
||||||
let mutations = {}
|
|
||||||
$columns.forEach(column => {
|
|
||||||
mutations[column.name] = { visible }
|
|
||||||
})
|
|
||||||
datasource.actions.addSchemaMutations(mutations)
|
|
||||||
await datasource.actions.saveSchemaMutations()
|
await datasource.actions.saveSchemaMutations()
|
||||||
dispatch(visible ? "show-column" : "hide-column")
|
dispatch(visible ? "show-column" : "hide-column")
|
||||||
}
|
}
|
||||||
|
|
||||||
const getText = columns => {
|
const getText = columns => {
|
||||||
const hidden = columns.filter(col => !col.visible).length
|
const hidden = columns.filter(col => !col.visible).length
|
||||||
return hidden ? `Hide columns (${hidden})` : "Hide columns"
|
return hidden ? `Columns (${hidden} restricted)` : "Columns"
|
||||||
|
}
|
||||||
|
|
||||||
|
const PERMISSION_OPTIONS = {
|
||||||
|
WRITABLE: "writable",
|
||||||
|
HIDDEN: "hidden",
|
||||||
|
}
|
||||||
|
|
||||||
|
const options = [
|
||||||
|
{ icon: "Edit", value: PERMISSION_OPTIONS.WRITABLE, tooltip: "Writable" },
|
||||||
|
{
|
||||||
|
icon: "VisibilityOff",
|
||||||
|
value: PERMISSION_OPTIONS.HIDDEN,
|
||||||
|
tooltip: "Hidden",
|
||||||
|
},
|
||||||
|
]
|
||||||
|
|
||||||
|
function columnToPermissionOptions(column) {
|
||||||
|
if (!column.visible) {
|
||||||
|
return PERMISSION_OPTIONS.HIDDEN
|
||||||
|
}
|
||||||
|
|
||||||
|
return PERMISSION_OPTIONS.WRITABLE
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div bind:this={anchor}>
|
<div bind:this={anchor}>
|
||||||
<ActionButton
|
<ActionButton
|
||||||
icon="VisibilityOff"
|
icon="ColumnSettings"
|
||||||
quiet
|
quiet
|
||||||
size="M"
|
size="M"
|
||||||
on:click={() => (open = !open)}
|
on:click={() => (open = !open)}
|
||||||
|
@ -54,25 +69,25 @@
|
||||||
<Icon size="S" name={getColumnIcon($stickyColumn)} />
|
<Icon size="S" name={getColumnIcon($stickyColumn)} />
|
||||||
{$stickyColumn.label}
|
{$stickyColumn.label}
|
||||||
</div>
|
</div>
|
||||||
<Toggle disabled size="S" value={true} />
|
|
||||||
|
<ToggleActionButtonGroup
|
||||||
|
disabled
|
||||||
|
value={PERMISSION_OPTIONS.WRITABLE}
|
||||||
|
{options}
|
||||||
|
/>
|
||||||
{/if}
|
{/if}
|
||||||
{#each $columns as column}
|
{#each $columns as column}
|
||||||
<div class="column">
|
<div class="column">
|
||||||
<Icon size="S" name={getColumnIcon(column)} />
|
<Icon size="S" name={getColumnIcon(column)} />
|
||||||
{column.label}
|
{column.label}
|
||||||
</div>
|
</div>
|
||||||
<Toggle
|
<ToggleActionButtonGroup
|
||||||
size="S"
|
on:click={e => toggleColumn(column, e.detail)}
|
||||||
value={column.visible}
|
value={columnToPermissionOptions(column)}
|
||||||
on:change={e => toggleColumn(column, e.detail)}
|
{options}
|
||||||
disabled={column.primaryDisplay}
|
|
||||||
/>
|
/>
|
||||||
{/each}
|
{/each}
|
||||||
</div>
|
</div>
|
||||||
<div class="buttons">
|
|
||||||
<ActionButton on:click={() => toggleAll(true)}>Show all</ActionButton>
|
|
||||||
<ActionButton on:click={() => toggleAll(false)}>Hide all</ActionButton>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
</Popover>
|
</Popover>
|
||||||
|
|
||||||
|
@ -83,15 +98,11 @@
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
gap: 12px;
|
gap: 12px;
|
||||||
}
|
}
|
||||||
.buttons {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: row;
|
|
||||||
gap: 8px;
|
|
||||||
}
|
|
||||||
.columns {
|
.columns {
|
||||||
display: grid;
|
display: grid;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
grid-template-columns: 1fr auto;
|
grid-template-columns: 1fr auto;
|
||||||
|
gap: 8px;
|
||||||
}
|
}
|
||||||
.columns :global(.spectrum-Switch) {
|
.columns :global(.spectrum-Switch) {
|
||||||
margin-right: 0;
|
margin-right: 0;
|
|
@ -0,0 +1,43 @@
|
||||||
|
<script>
|
||||||
|
import { createEventDispatcher } from "svelte"
|
||||||
|
|
||||||
|
let dispatch = createEventDispatcher()
|
||||||
|
|
||||||
|
import { ActionButton, AbsTooltip, TooltipType } from "@budibase/bbui"
|
||||||
|
|
||||||
|
export let value
|
||||||
|
export let options
|
||||||
|
export let disabled
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<div class="permissionPicker">
|
||||||
|
{#each options as option}
|
||||||
|
<AbsTooltip text={option.tooltip} type={TooltipType.Info}>
|
||||||
|
<ActionButton
|
||||||
|
on:click={() => dispatch("click", option.value)}
|
||||||
|
{disabled}
|
||||||
|
size="S"
|
||||||
|
icon={option.icon}
|
||||||
|
quiet
|
||||||
|
selected={option.value === value}
|
||||||
|
noPadding
|
||||||
|
/>
|
||||||
|
</AbsTooltip>
|
||||||
|
{/each}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.permissionPicker {
|
||||||
|
display: flex;
|
||||||
|
gap: var(--spacing-xs);
|
||||||
|
padding-left: calc(var(--spacing-xl) * 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.permissionPicker :global(.spectrum-Icon) {
|
||||||
|
width: 14px;
|
||||||
|
}
|
||||||
|
.permissionPicker :global(.spectrum-ActionButton) {
|
||||||
|
width: 28px;
|
||||||
|
height: 28px;
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -16,9 +16,10 @@
|
||||||
scroll,
|
scroll,
|
||||||
isDragging,
|
isDragging,
|
||||||
buttonColumnWidth,
|
buttonColumnWidth,
|
||||||
|
showVScrollbar,
|
||||||
} = getContext("grid")
|
} = getContext("grid")
|
||||||
|
|
||||||
let measureContainer
|
let container
|
||||||
|
|
||||||
$: buttons = $props.buttons?.slice(0, 3) || []
|
$: buttons = $props.buttons?.slice(0, 3) || []
|
||||||
$: columnsWidth = $visibleColumns.reduce(
|
$: columnsWidth = $visibleColumns.reduce(
|
||||||
|
@ -39,7 +40,7 @@
|
||||||
const width = entries?.[0]?.contentRect?.width ?? 0
|
const width = entries?.[0]?.contentRect?.width ?? 0
|
||||||
buttonColumnWidth.set(width)
|
buttonColumnWidth.set(width)
|
||||||
})
|
})
|
||||||
observer.observe(measureContainer)
|
observer.observe(container)
|
||||||
})
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
@ -50,7 +51,7 @@
|
||||||
class:hidden={$buttonColumnWidth === 0}
|
class:hidden={$buttonColumnWidth === 0}
|
||||||
>
|
>
|
||||||
<div class="content" on:mouseleave={() => ($hoveredRowId = null)}>
|
<div class="content" on:mouseleave={() => ($hoveredRowId = null)}>
|
||||||
<GridScrollWrapper scrollVertically attachHandlers>
|
<GridScrollWrapper scrollVertically attachHandlers bind:ref={container}>
|
||||||
{#each $renderedRows as row}
|
{#each $renderedRows as row}
|
||||||
{@const rowSelected = !!$selectedRows[row._id]}
|
{@const rowSelected = !!$selectedRows[row._id]}
|
||||||
{@const rowHovered = $hoveredRowId === row._id}
|
{@const rowHovered = $hoveredRowId === row._id}
|
||||||
|
@ -59,7 +60,6 @@
|
||||||
class="row"
|
class="row"
|
||||||
on:mouseenter={$isDragging ? null : () => ($hoveredRowId = row._id)}
|
on:mouseenter={$isDragging ? null : () => ($hoveredRowId = row._id)}
|
||||||
on:mouseleave={$isDragging ? null : () => ($hoveredRowId = null)}
|
on:mouseleave={$isDragging ? null : () => ($hoveredRowId = null)}
|
||||||
bind:this={measureContainer}
|
|
||||||
>
|
>
|
||||||
<GridCell
|
<GridCell
|
||||||
width="auto"
|
width="auto"
|
||||||
|
@ -67,7 +67,7 @@
|
||||||
selected={rowSelected}
|
selected={rowSelected}
|
||||||
highlighted={rowHovered || rowFocused}
|
highlighted={rowHovered || rowFocused}
|
||||||
>
|
>
|
||||||
<div class="buttons">
|
<div class="buttons" class:offset={$showVScrollbar}>
|
||||||
{#each buttons as button}
|
{#each buttons as button}
|
||||||
<Button
|
<Button
|
||||||
newStyles
|
newStyles
|
||||||
|
@ -121,6 +121,9 @@
|
||||||
gap: var(--cell-padding);
|
gap: var(--cell-padding);
|
||||||
height: inherit;
|
height: inherit;
|
||||||
}
|
}
|
||||||
|
.buttons.offset {
|
||||||
|
padding-right: calc(var(--cell-padding) + 2 * var(--scroll-bar-size) - 2px);
|
||||||
|
}
|
||||||
.buttons :global(.spectrum-Button-Label) {
|
.buttons :global(.spectrum-Button-Label) {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
|
|
|
@ -18,7 +18,7 @@
|
||||||
import UserAvatars from "./UserAvatars.svelte"
|
import UserAvatars from "./UserAvatars.svelte"
|
||||||
import KeyboardManager from "../overlays/KeyboardManager.svelte"
|
import KeyboardManager from "../overlays/KeyboardManager.svelte"
|
||||||
import SortButton from "../controls/SortButton.svelte"
|
import SortButton from "../controls/SortButton.svelte"
|
||||||
import HideColumnsButton from "../controls/HideColumnsButton.svelte"
|
import ColumnsSettingButton from "../controls/ColumnsSettingButton.svelte"
|
||||||
import SizeButton from "../controls/SizeButton.svelte"
|
import SizeButton from "../controls/SizeButton.svelte"
|
||||||
import NewRow from "./NewRow.svelte"
|
import NewRow from "./NewRow.svelte"
|
||||||
import { createGridWebsocket } from "../lib/websocket"
|
import { createGridWebsocket } from "../lib/websocket"
|
||||||
|
@ -29,6 +29,7 @@
|
||||||
Padding,
|
Padding,
|
||||||
SmallRowHeight,
|
SmallRowHeight,
|
||||||
ControlsHeight,
|
ControlsHeight,
|
||||||
|
ScrollBarSize,
|
||||||
} from "../lib/constants"
|
} from "../lib/constants"
|
||||||
|
|
||||||
export let API = null
|
export let API = null
|
||||||
|
@ -145,14 +146,14 @@
|
||||||
class:quiet
|
class:quiet
|
||||||
on:mouseenter={() => gridFocused.set(true)}
|
on:mouseenter={() => gridFocused.set(true)}
|
||||||
on:mouseleave={() => gridFocused.set(false)}
|
on:mouseleave={() => gridFocused.set(false)}
|
||||||
style="--row-height:{$rowHeight}px; --default-row-height:{DefaultRowHeight}px; --gutter-width:{GutterWidth}px; --max-cell-render-overflow:{MaxCellRenderOverflow}px; --content-lines:{$contentLines}; --min-height:{$minHeight}px; --controls-height:{ControlsHeight}px;"
|
style="--row-height:{$rowHeight}px; --default-row-height:{DefaultRowHeight}px; --gutter-width:{GutterWidth}px; --max-cell-render-overflow:{MaxCellRenderOverflow}px; --content-lines:{$contentLines}; --min-height:{$minHeight}px; --controls-height:{ControlsHeight}px; --scroll-bar-size:{ScrollBarSize}px;"
|
||||||
>
|
>
|
||||||
{#if showControls}
|
{#if showControls}
|
||||||
<div class="controls">
|
<div class="controls">
|
||||||
<div class="controls-left">
|
<div class="controls-left">
|
||||||
<slot name="filter" />
|
<slot name="filter" />
|
||||||
<SortButton />
|
<SortButton />
|
||||||
<HideColumnsButton />
|
<ColumnsSettingButton />
|
||||||
<SizeButton />
|
<SizeButton />
|
||||||
<slot name="controls" />
|
<slot name="controls" />
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -18,6 +18,7 @@
|
||||||
export let scrollVertically = false
|
export let scrollVertically = false
|
||||||
export let scrollHorizontally = false
|
export let scrollHorizontally = false
|
||||||
export let attachHandlers = false
|
export let attachHandlers = false
|
||||||
|
export let ref
|
||||||
|
|
||||||
// Used for tracking touch events
|
// Used for tracking touch events
|
||||||
let initialTouchX
|
let initialTouchX
|
||||||
|
@ -109,7 +110,7 @@
|
||||||
on:touchmove={attachHandlers ? handleTouchMove : null}
|
on:touchmove={attachHandlers ? handleTouchMove : null}
|
||||||
on:click|self={() => ($focusedCellId = null)}
|
on:click|self={() => ($focusedCellId = null)}
|
||||||
>
|
>
|
||||||
<div {style} class="inner">
|
<div {style} class="inner" bind:this={ref}>
|
||||||
<slot />
|
<slot />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -119,7 +119,7 @@
|
||||||
<!-- svelte-ignore a11y-no-static-element-interactions -->
|
<!-- svelte-ignore a11y-no-static-element-interactions -->
|
||||||
<div
|
<div
|
||||||
class="v-scrollbar"
|
class="v-scrollbar"
|
||||||
style="--size:{ScrollBarSize}px; top:{barTop}px; height:{barHeight}px;"
|
style="top:{barTop}px; height:{barHeight}px;"
|
||||||
on:mousedown={startVDragging}
|
on:mousedown={startVDragging}
|
||||||
on:touchstart={startVDragging}
|
on:touchstart={startVDragging}
|
||||||
class:dragging={isDraggingV}
|
class:dragging={isDraggingV}
|
||||||
|
@ -129,7 +129,7 @@
|
||||||
<!-- svelte-ignore a11y-no-static-element-interactions -->
|
<!-- svelte-ignore a11y-no-static-element-interactions -->
|
||||||
<div
|
<div
|
||||||
class="h-scrollbar"
|
class="h-scrollbar"
|
||||||
style="--size:{ScrollBarSize}px; left:{barLeft}px; width:{barWidth}px;"
|
style="left:{barLeft}px; width:{barWidth}px;"
|
||||||
on:mousedown={startHDragging}
|
on:mousedown={startHDragging}
|
||||||
on:touchstart={startHDragging}
|
on:touchstart={startHDragging}
|
||||||
class:dragging={isDraggingH}
|
class:dragging={isDraggingH}
|
||||||
|
@ -149,11 +149,11 @@
|
||||||
opacity: 1;
|
opacity: 1;
|
||||||
}
|
}
|
||||||
.v-scrollbar {
|
.v-scrollbar {
|
||||||
width: var(--size);
|
width: var(--scroll-bar-size);
|
||||||
right: var(--size);
|
right: var(--scroll-bar-size);
|
||||||
}
|
}
|
||||||
.h-scrollbar {
|
.h-scrollbar {
|
||||||
height: var(--size);
|
height: var(--scroll-bar-size);
|
||||||
bottom: var(--size);
|
bottom: var(--scroll-bar-size);
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -404,8 +404,11 @@ export const createActions = context => {
|
||||||
|
|
||||||
// Save change
|
// Save change
|
||||||
try {
|
try {
|
||||||
// Mark as in progress
|
// Increment change count for this row
|
||||||
inProgressChanges.update(state => ({ ...state, [rowId]: true }))
|
inProgressChanges.update(state => ({
|
||||||
|
...state,
|
||||||
|
[rowId]: (state[rowId] || 0) + 1,
|
||||||
|
}))
|
||||||
|
|
||||||
// Update row
|
// Update row
|
||||||
const changes = get(rowChangeCache)[rowId]
|
const changes = get(rowChangeCache)[rowId]
|
||||||
|
@ -423,17 +426,25 @@ export const createActions = context => {
|
||||||
await refreshRow(saved.id)
|
await refreshRow(saved.id)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Wipe row change cache now that we've saved the row
|
// Wipe row change cache for any values which have been saved
|
||||||
|
const liveChanges = get(rowChangeCache)[rowId]
|
||||||
rowChangeCache.update(state => {
|
rowChangeCache.update(state => {
|
||||||
delete state[rowId]
|
Object.keys(changes || {}).forEach(key => {
|
||||||
|
if (changes[key] === liveChanges?.[key]) {
|
||||||
|
delete state[rowId][key]
|
||||||
|
}
|
||||||
|
})
|
||||||
return state
|
return state
|
||||||
})
|
})
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
handleValidationError(rowId, error)
|
handleValidationError(rowId, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Mark as completed
|
// Decrement change count for this row
|
||||||
inProgressChanges.update(state => ({ ...state, [rowId]: false }))
|
inProgressChanges.update(state => ({
|
||||||
|
...state,
|
||||||
|
[rowId]: (state[rowId] || 1) - 1,
|
||||||
|
}))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Updates a value of a row
|
// Updates a value of a row
|
||||||
|
@ -553,7 +564,6 @@ export const initialise = context => {
|
||||||
previousFocusedCellId,
|
previousFocusedCellId,
|
||||||
rows,
|
rows,
|
||||||
validation,
|
validation,
|
||||||
focusedCellId,
|
|
||||||
} = context
|
} = context
|
||||||
|
|
||||||
// Wipe the row change cache when changing row
|
// Wipe the row change cache when changing row
|
||||||
|
@ -571,20 +581,12 @@ export const initialise = context => {
|
||||||
if (!id) {
|
if (!id) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
// Stop if we changed row
|
const { id: rowId, field } = parseCellID(id)
|
||||||
const split = parseCellID(id)
|
const hasChanges = field in (get(rowChangeCache)[rowId] || {})
|
||||||
const oldRowId = split.id
|
const hasErrors = validation.actions.rowHasErrors(rowId)
|
||||||
const oldColumn = split.field
|
const isSavingChanges = get(inProgressChanges)[rowId]
|
||||||
const { id: newRowId } = parseCellID(get(focusedCellId))
|
if (rowId && !hasErrors && hasChanges && !isSavingChanges) {
|
||||||
if (oldRowId !== newRowId) {
|
await rows.actions.applyRowChanges(rowId)
|
||||||
return
|
|
||||||
}
|
|
||||||
// Otherwise we just changed cell in the same row
|
|
||||||
const hasChanges = oldColumn in (get(rowChangeCache)[oldRowId] || {})
|
|
||||||
const hasErrors = validation.actions.rowHasErrors(oldRowId)
|
|
||||||
const isSavingChanges = get(inProgressChanges)[oldRowId]
|
|
||||||
if (oldRowId && !hasErrors && hasChanges && !isSavingChanges) {
|
|
||||||
await rows.actions.applyRowChanges(oldRowId)
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
|
@ -109,6 +109,7 @@ export const initialise = context => {
|
||||||
maxScrollTop,
|
maxScrollTop,
|
||||||
scrollLeft,
|
scrollLeft,
|
||||||
maxScrollLeft,
|
maxScrollLeft,
|
||||||
|
buttonColumnWidth,
|
||||||
} = context
|
} = context
|
||||||
|
|
||||||
// Ensure scroll state never goes invalid, which can happen when changing
|
// Ensure scroll state never goes invalid, which can happen when changing
|
||||||
|
@ -194,8 +195,10 @@ export const initialise = context => {
|
||||||
|
|
||||||
// Ensure column is not cutoff on right edge
|
// Ensure column is not cutoff on right edge
|
||||||
else {
|
else {
|
||||||
|
const $buttonColumnWidth = get(buttonColumnWidth)
|
||||||
const rightEdge = column.left + column.width
|
const rightEdge = column.left + column.width
|
||||||
const rightBound = $bounds.width + $scroll.left - FocusedCellMinOffset
|
const rightBound =
|
||||||
|
$bounds.width + $scroll.left - FocusedCellMinOffset - $buttonColumnWidth
|
||||||
delta = rightEdge - rightBound
|
delta = rightEdge - rightBound
|
||||||
if (delta > 0) {
|
if (delta > 0) {
|
||||||
scroll.update(state => ({
|
scroll.update(state => ({
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
Subproject commit 5189b83bea1868574ff7f4c51fe5db38a11badb8
|
Subproject commit d3c3077011a8e20ed3c48dcd6301caca4120b6ac
|
|
@ -68,7 +68,6 @@
|
||||||
"aws-sdk": "2.1030.0",
|
"aws-sdk": "2.1030.0",
|
||||||
"bcrypt": "5.1.0",
|
"bcrypt": "5.1.0",
|
||||||
"bcryptjs": "2.4.3",
|
"bcryptjs": "2.4.3",
|
||||||
"bl": "^6.0.12",
|
|
||||||
"bull": "4.10.1",
|
"bull": "4.10.1",
|
||||||
"chokidar": "3.5.3",
|
"chokidar": "3.5.3",
|
||||||
"content-disposition": "^0.5.4",
|
"content-disposition": "^0.5.4",
|
||||||
|
@ -97,7 +96,7 @@
|
||||||
"memorystream": "0.3.1",
|
"memorystream": "0.3.1",
|
||||||
"mongodb": "^6.3.0",
|
"mongodb": "^6.3.0",
|
||||||
"mssql": "10.0.1",
|
"mssql": "10.0.1",
|
||||||
"mysql2": "3.9.7",
|
"mysql2": "3.9.8",
|
||||||
"node-fetch": "2.6.7",
|
"node-fetch": "2.6.7",
|
||||||
"object-sizeof": "2.6.1",
|
"object-sizeof": "2.6.1",
|
||||||
"openai": "^3.2.1",
|
"openai": "^3.2.1",
|
||||||
|
@ -116,7 +115,8 @@
|
||||||
"uuid": "^8.3.2",
|
"uuid": "^8.3.2",
|
||||||
"validate.js": "0.13.1",
|
"validate.js": "0.13.1",
|
||||||
"worker-farm": "1.7.0",
|
"worker-farm": "1.7.0",
|
||||||
"xml2js": "0.5.0"
|
"xml2js": "0.5.0",
|
||||||
|
"tmp": "0.2.3"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@babel/preset-env": "7.16.11",
|
"@babel/preset-env": "7.16.11",
|
||||||
|
@ -137,6 +137,7 @@
|
||||||
"@types/supertest": "2.0.14",
|
"@types/supertest": "2.0.14",
|
||||||
"@types/tar": "6.1.5",
|
"@types/tar": "6.1.5",
|
||||||
"@types/uuid": "8.3.4",
|
"@types/uuid": "8.3.4",
|
||||||
|
"@types/tmp": "0.2.6",
|
||||||
"copyfiles": "2.4.1",
|
"copyfiles": "2.4.1",
|
||||||
"docker-compose": "0.23.17",
|
"docker-compose": "0.23.17",
|
||||||
"jest": "29.7.0",
|
"jest": "29.7.0",
|
||||||
|
|
Binary file not shown.
|
@ -1,24 +0,0 @@
|
||||||
const setup = require("./utilities")
|
|
||||||
|
|
||||||
describe("/templates", () => {
|
|
||||||
let request = setup.getRequest()
|
|
||||||
let config = setup.getConfig()
|
|
||||||
|
|
||||||
afterAll(setup.afterAll)
|
|
||||||
|
|
||||||
beforeAll(async () => {
|
|
||||||
await config.init()
|
|
||||||
})
|
|
||||||
|
|
||||||
describe("fetch", () => {
|
|
||||||
it("should be able to fetch templates", async () => {
|
|
||||||
const res = await request
|
|
||||||
.get(`/api/templates`)
|
|
||||||
.set(config.defaultHeaders())
|
|
||||||
.expect("Content-Type", /json/)
|
|
||||||
.expect(200)
|
|
||||||
// this test is quite light right now, templates aren't heavily utilised yet
|
|
||||||
expect(Array.isArray(res.body)).toEqual(true)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
})
|
|
|
@ -0,0 +1,125 @@
|
||||||
|
import * as setup from "./utilities"
|
||||||
|
import path from "path"
|
||||||
|
import nock from "nock"
|
||||||
|
import { generator } from "@budibase/backend-core/tests"
|
||||||
|
|
||||||
|
interface App {
|
||||||
|
background: string
|
||||||
|
icon: string
|
||||||
|
category: string
|
||||||
|
description: string
|
||||||
|
name: string
|
||||||
|
url: string
|
||||||
|
type: string
|
||||||
|
key: string
|
||||||
|
image: string
|
||||||
|
}
|
||||||
|
|
||||||
|
interface Manifest {
|
||||||
|
templates: {
|
||||||
|
app: { [key: string]: App }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function setManifest(manifest: Manifest) {
|
||||||
|
nock("https://prod-budi-templates.s3-eu-west-1.amazonaws.com")
|
||||||
|
.get("/manifest.json")
|
||||||
|
.reply(200, manifest)
|
||||||
|
}
|
||||||
|
|
||||||
|
function mockApp(key: string, tarPath: string) {
|
||||||
|
nock("https://prod-budi-templates.s3-eu-west-1.amazonaws.com")
|
||||||
|
.get(`/templates/app/${key}.tar.gz`)
|
||||||
|
.replyWithFile(200, tarPath)
|
||||||
|
}
|
||||||
|
|
||||||
|
function mockAgencyClientPortal() {
|
||||||
|
setManifest({
|
||||||
|
templates: {
|
||||||
|
app: {
|
||||||
|
"Agency Client Portal": {
|
||||||
|
background: "#20a3a8",
|
||||||
|
icon: "Project",
|
||||||
|
category: "Portals",
|
||||||
|
description:
|
||||||
|
"Manage clients, streamline communications, and securely share files.",
|
||||||
|
name: "Agency Client Portal",
|
||||||
|
url: "https://budibase.com/portals/templates/agency-client-portal-template/",
|
||||||
|
type: "app",
|
||||||
|
key: "app/agency-client-portal",
|
||||||
|
image:
|
||||||
|
"https://prod-budi-templates.s3.eu-west-1.amazonaws.com/images/agency-client-portal.png",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
mockApp(
|
||||||
|
"agency-client-portal",
|
||||||
|
path.resolve(__dirname, "data", "agency-client-portal.tar.gz")
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
describe("/templates", () => {
|
||||||
|
let config = setup.getConfig()
|
||||||
|
|
||||||
|
afterAll(setup.afterAll)
|
||||||
|
beforeAll(async () => {
|
||||||
|
await config.init()
|
||||||
|
})
|
||||||
|
beforeEach(() => {
|
||||||
|
nock.cleanAll()
|
||||||
|
mockAgencyClientPortal()
|
||||||
|
})
|
||||||
|
|
||||||
|
describe("fetch", () => {
|
||||||
|
it("should be able to fetch templates", async () => {
|
||||||
|
const templates = await config.api.templates.fetch()
|
||||||
|
expect(templates).toHaveLength(1)
|
||||||
|
expect(templates[0].name).toBe("Agency Client Portal")
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe("create app from template", () => {
|
||||||
|
it.each(["sqs", "lucene"])(
|
||||||
|
`should be able to create an app from a template (%s)`,
|
||||||
|
async source => {
|
||||||
|
const env = {
|
||||||
|
SQS_SEARCH_ENABLE: source === "sqs" ? "true" : "false",
|
||||||
|
}
|
||||||
|
|
||||||
|
await config.withEnv(env, async () => {
|
||||||
|
const name = generator.guid().replaceAll("-", "")
|
||||||
|
const url = `/${name}`
|
||||||
|
|
||||||
|
const app = await config.api.application.create({
|
||||||
|
name,
|
||||||
|
url,
|
||||||
|
useTemplate: "true",
|
||||||
|
templateName: "Agency Client Portal",
|
||||||
|
templateKey: "app/agency-client-portal",
|
||||||
|
})
|
||||||
|
expect(app.name).toBe(name)
|
||||||
|
expect(app.url).toBe(url)
|
||||||
|
|
||||||
|
await config.withApp(app, async () => {
|
||||||
|
const tables = await config.api.table.fetch()
|
||||||
|
expect(tables).toHaveLength(2)
|
||||||
|
|
||||||
|
tables.sort((a, b) => a.name.localeCompare(b.name))
|
||||||
|
const [agencyProjects, users] = tables
|
||||||
|
expect(agencyProjects.name).toBe("Agency Projects")
|
||||||
|
expect(users.name).toBe("Users")
|
||||||
|
|
||||||
|
const { rows } = await config.api.row.search(agencyProjects._id!, {
|
||||||
|
tableId: agencyProjects._id!,
|
||||||
|
query: {},
|
||||||
|
})
|
||||||
|
|
||||||
|
expect(rows).toHaveLength(3)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
)
|
||||||
|
})
|
||||||
|
})
|
|
@ -149,13 +149,12 @@ class RestIntegration implements IntegrationBase {
|
||||||
{ downloadImages: this.config.downloadImages }
|
{ downloadImages: this.config.downloadImages }
|
||||||
)
|
)
|
||||||
let contentLength = response.headers.get("content-length")
|
let contentLength = response.headers.get("content-length")
|
||||||
if (!contentLength && raw) {
|
let isSuccess = response.status >= 200 && response.status < 300
|
||||||
contentLength = Buffer.byteLength(raw, "utf8").toString()
|
|
||||||
}
|
|
||||||
if (
|
if (
|
||||||
contentDisposition.includes("filename") ||
|
(contentDisposition.includes("filename") ||
|
||||||
contentDisposition.includes("attachment") ||
|
contentDisposition.includes("attachment") ||
|
||||||
contentDisposition.includes("form-data")
|
contentDisposition.includes("form-data")) &&
|
||||||
|
isSuccess
|
||||||
) {
|
) {
|
||||||
filename =
|
filename =
|
||||||
path.basename(parse(contentDisposition).parameters?.filename) || ""
|
path.basename(parse(contentDisposition).parameters?.filename) || ""
|
||||||
|
@ -168,6 +167,9 @@ class RestIntegration implements IntegrationBase {
|
||||||
return handleFileResponse(response, filename, this.startTimeMs)
|
return handleFileResponse(response, filename, this.startTimeMs)
|
||||||
} else {
|
} else {
|
||||||
responseTxt = response.text ? await response.text() : ""
|
responseTxt = response.text ? await response.text() : ""
|
||||||
|
if (!contentLength && responseTxt) {
|
||||||
|
contentLength = Buffer.byteLength(responseTxt, "utf8").toString()
|
||||||
|
}
|
||||||
const hasContent =
|
const hasContent =
|
||||||
(contentLength && parseInt(contentLength) > 0) ||
|
(contentLength && parseInt(contentLength) > 0) ||
|
||||||
responseTxt.length > 0
|
responseTxt.length > 0
|
||||||
|
|
|
@ -657,6 +657,7 @@ describe("REST Integration", () => {
|
||||||
mockReadable.push(null)
|
mockReadable.push(null)
|
||||||
;(fetch as unknown as jest.Mock).mockImplementationOnce(() =>
|
;(fetch as unknown as jest.Mock).mockImplementationOnce(() =>
|
||||||
Promise.resolve({
|
Promise.resolve({
|
||||||
|
status: 200,
|
||||||
headers: {
|
headers: {
|
||||||
raw: () => ({
|
raw: () => ({
|
||||||
"content-type": [contentType],
|
"content-type": [contentType],
|
||||||
|
@ -700,6 +701,7 @@ describe("REST Integration", () => {
|
||||||
mockReadable.push(null)
|
mockReadable.push(null)
|
||||||
;(fetch as unknown as jest.Mock).mockImplementationOnce(() =>
|
;(fetch as unknown as jest.Mock).mockImplementationOnce(() =>
|
||||||
Promise.resolve({
|
Promise.resolve({
|
||||||
|
status: 200,
|
||||||
headers: {
|
headers: {
|
||||||
raw: () => ({
|
raw: () => ({
|
||||||
"content-type": [contentType],
|
"content-type": [contentType],
|
||||||
|
|
|
@ -18,7 +18,7 @@ class MariaDBWaitStrategy extends AbstractWaitStrategy {
|
||||||
await logs.waitUntilReady(container, boundPorts, startTime)
|
await logs.waitUntilReady(container, boundPorts, startTime)
|
||||||
|
|
||||||
const command = Wait.forSuccessfulCommand(
|
const command = Wait.forSuccessfulCommand(
|
||||||
`mysqladmin ping -h localhost -P 3306 -u root -ppassword`
|
`/usr/local/bin/healthcheck.sh --innodb_initialized`
|
||||||
)
|
)
|
||||||
await command.waitUntilReady(container)
|
await command.waitUntilReady(container)
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,10 +9,12 @@ import { context, objectStore, sql } from "@budibase/backend-core"
|
||||||
import { v4 } from "uuid"
|
import { v4 } from "uuid"
|
||||||
import { parseStringPromise as xmlParser } from "xml2js"
|
import { parseStringPromise as xmlParser } from "xml2js"
|
||||||
import { formatBytes } from "../../utilities"
|
import { formatBytes } from "../../utilities"
|
||||||
import bl from "bl"
|
|
||||||
import env from "../../environment"
|
import env from "../../environment"
|
||||||
import { InvalidColumns } from "../../constants"
|
import { InvalidColumns } from "../../constants"
|
||||||
import { helpers, utils } from "@budibase/shared-core"
|
import { helpers, utils } from "@budibase/shared-core"
|
||||||
|
import { pipeline } from "stream/promises"
|
||||||
|
import tmp from "tmp"
|
||||||
|
import fs from "fs"
|
||||||
|
|
||||||
type PrimitiveTypes =
|
type PrimitiveTypes =
|
||||||
| FieldType.STRING
|
| FieldType.STRING
|
||||||
|
@ -360,35 +362,44 @@ export async function handleFileResponse(
|
||||||
const key = `${context.getProdAppId()}/${processedFileName}`
|
const key = `${context.getProdAppId()}/${processedFileName}`
|
||||||
const bucket = objectStore.ObjectStoreBuckets.TEMP
|
const bucket = objectStore.ObjectStoreBuckets.TEMP
|
||||||
|
|
||||||
const stream = response.body.pipe(bl((error, data) => data))
|
// put the response stream to disk temporarily as a buffer
|
||||||
|
const tmpObj = tmp.fileSync()
|
||||||
|
try {
|
||||||
|
await pipeline(response.body, fs.createWriteStream(tmpObj.name))
|
||||||
|
if (response.body) {
|
||||||
|
const contentLength = response.headers.get("content-length")
|
||||||
|
if (contentLength) {
|
||||||
|
size = parseInt(contentLength, 10)
|
||||||
|
}
|
||||||
|
|
||||||
if (response.body) {
|
const details = await objectStore.streamUpload({
|
||||||
const contentLength = response.headers.get("content-length")
|
bucket,
|
||||||
if (contentLength) {
|
filename: key,
|
||||||
size = parseInt(contentLength, 10)
|
stream: fs.createReadStream(tmpObj.name),
|
||||||
|
ttl: 1,
|
||||||
|
type: response.headers["content-type"],
|
||||||
|
})
|
||||||
|
if (!size && details.ContentLength) {
|
||||||
|
size = details.ContentLength
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
presignedUrl = objectStore.getPresignedUrl(bucket, key)
|
||||||
await objectStore.streamUpload({
|
return {
|
||||||
bucket,
|
data: {
|
||||||
filename: key,
|
size,
|
||||||
stream,
|
name: processedFileName,
|
||||||
ttl: 1,
|
url: presignedUrl,
|
||||||
type: response.headers["content-type"],
|
extension: fileExtension,
|
||||||
})
|
key: key,
|
||||||
}
|
},
|
||||||
presignedUrl = objectStore.getPresignedUrl(bucket, key)
|
info: {
|
||||||
return {
|
code: response.status,
|
||||||
data: {
|
size: formatBytes(size.toString()),
|
||||||
size,
|
time: `${Math.round(performance.now() - startTime)}ms`,
|
||||||
name: processedFileName,
|
},
|
||||||
url: presignedUrl,
|
}
|
||||||
extension: fileExtension,
|
} finally {
|
||||||
key: key,
|
// cleanup tmp
|
||||||
},
|
tmpObj.removeCallback()
|
||||||
info: {
|
|
||||||
code: response.status,
|
|
||||||
size: formatBytes(size.toString()),
|
|
||||||
time: `${Math.round(performance.now() - startTime)}ms`,
|
|
||||||
},
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -314,6 +314,16 @@ export default class TestConfiguration {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async withApp(app: App | string, f: () => Promise<void>) {
|
||||||
|
const oldAppId = this.appId
|
||||||
|
this.appId = typeof app === "string" ? app : app.appId
|
||||||
|
try {
|
||||||
|
return await f()
|
||||||
|
} finally {
|
||||||
|
this.appId = oldAppId
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// UTILS
|
// UTILS
|
||||||
|
|
||||||
_req<Req extends Record<string, any> | void, Res>(
|
_req<Req extends Record<string, any> | void, Res>(
|
||||||
|
|
|
@ -12,6 +12,7 @@ import { AttachmentAPI } from "./attachment"
|
||||||
import { UserAPI } from "./user"
|
import { UserAPI } from "./user"
|
||||||
import { QueryAPI } from "./query"
|
import { QueryAPI } from "./query"
|
||||||
import { RoleAPI } from "./role"
|
import { RoleAPI } from "./role"
|
||||||
|
import { TemplateAPI } from "./template"
|
||||||
|
|
||||||
export default class API {
|
export default class API {
|
||||||
table: TableAPI
|
table: TableAPI
|
||||||
|
@ -27,6 +28,7 @@ export default class API {
|
||||||
user: UserAPI
|
user: UserAPI
|
||||||
query: QueryAPI
|
query: QueryAPI
|
||||||
roles: RoleAPI
|
roles: RoleAPI
|
||||||
|
templates: TemplateAPI
|
||||||
|
|
||||||
constructor(config: TestConfiguration) {
|
constructor(config: TestConfiguration) {
|
||||||
this.table = new TableAPI(config)
|
this.table = new TableAPI(config)
|
||||||
|
@ -42,5 +44,6 @@ export default class API {
|
||||||
this.user = new UserAPI(config)
|
this.user = new UserAPI(config)
|
||||||
this.query = new QueryAPI(config)
|
this.query = new QueryAPI(config)
|
||||||
this.roles = new RoleAPI(config)
|
this.roles = new RoleAPI(config)
|
||||||
|
this.templates = new TemplateAPI(config)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,8 @@
|
||||||
|
import { Template } from "@budibase/types"
|
||||||
|
import { Expectations, TestAPI } from "./base"
|
||||||
|
|
||||||
|
export class TemplateAPI extends TestAPI {
|
||||||
|
fetch = async (expectations?: Expectations): Promise<Template[]> => {
|
||||||
|
return await this._get<Template[]>("/api/templates", { expectations })
|
||||||
|
}
|
||||||
|
}
|
|
@ -196,12 +196,22 @@ class QueryRunner {
|
||||||
return { rows, keys, info, extra, pagination }
|
return { rows, keys, info, extra, pagination }
|
||||||
}
|
}
|
||||||
|
|
||||||
async runAnotherQuery(queryId: string, parameters: any) {
|
async runAnotherQuery(
|
||||||
|
queryId: string,
|
||||||
|
currentParameters: Record<string, any>
|
||||||
|
) {
|
||||||
const db = context.getAppDB()
|
const db = context.getAppDB()
|
||||||
const query = await db.get<Query>(queryId)
|
const query = await db.get<Query>(queryId)
|
||||||
const datasource = await sdk.datasources.get(query.datasourceId, {
|
const datasource = await sdk.datasources.get(query.datasourceId, {
|
||||||
enriched: true,
|
enriched: true,
|
||||||
})
|
})
|
||||||
|
// enrich parameters with dynamic queries defaults
|
||||||
|
const defaultParams = query.parameters || []
|
||||||
|
for (let param of defaultParams) {
|
||||||
|
if (!currentParameters[param.name]) {
|
||||||
|
currentParameters[param.name] = param.default
|
||||||
|
}
|
||||||
|
}
|
||||||
return new QueryRunner(
|
return new QueryRunner(
|
||||||
{
|
{
|
||||||
schema: query.schema,
|
schema: query.schema,
|
||||||
|
@ -210,7 +220,7 @@ class QueryRunner {
|
||||||
transformer: query.transformer,
|
transformer: query.transformer,
|
||||||
nullDefaultSupport: query.nullDefaultSupport,
|
nullDefaultSupport: query.nullDefaultSupport,
|
||||||
ctx: this.ctx,
|
ctx: this.ctx,
|
||||||
parameters,
|
parameters: currentParameters,
|
||||||
datasource,
|
datasource,
|
||||||
queryId,
|
queryId,
|
||||||
},
|
},
|
||||||
|
|
|
@ -45,7 +45,12 @@ export interface InviteUserRequest {
|
||||||
userInfo: any
|
userInfo: any
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface DeleteInviteUserRequest {
|
||||||
|
code: string
|
||||||
|
}
|
||||||
|
|
||||||
export type InviteUsersRequest = InviteUserRequest[]
|
export type InviteUsersRequest = InviteUserRequest[]
|
||||||
|
export type DeleteInviteUsersRequest = DeleteInviteUserRequest[]
|
||||||
|
|
||||||
export interface InviteUsersResponse {
|
export interface InviteUsersResponse {
|
||||||
successful: { email: string }[]
|
successful: { email: string }[]
|
||||||
|
|
|
@ -245,7 +245,7 @@ export type AutomationAttachment = {
|
||||||
|
|
||||||
export type AutomationAttachmentContent = {
|
export type AutomationAttachmentContent = {
|
||||||
filename: string
|
filename: string
|
||||||
content: ReadStream | NodeJS.ReadableStream | ReadableStream<Uint8Array>
|
content: ReadStream | NodeJS.ReadableStream
|
||||||
}
|
}
|
||||||
|
|
||||||
export type BucketedContent = AutomationAttachmentContent & {
|
export type BucketedContent = AutomationAttachmentContent & {
|
||||||
|
|
|
@ -10,6 +10,8 @@ import {
|
||||||
CreateAdminUserRequest,
|
CreateAdminUserRequest,
|
||||||
CreateAdminUserResponse,
|
CreateAdminUserResponse,
|
||||||
Ctx,
|
Ctx,
|
||||||
|
DeleteInviteUserRequest,
|
||||||
|
DeleteInviteUsersRequest,
|
||||||
InviteUserRequest,
|
InviteUserRequest,
|
||||||
InviteUsersRequest,
|
InviteUsersRequest,
|
||||||
InviteUsersResponse,
|
InviteUsersResponse,
|
||||||
|
@ -335,6 +337,20 @@ export const inviteMultiple = async (ctx: Ctx<InviteUsersRequest>) => {
|
||||||
ctx.body = await userSdk.invite(ctx.request.body)
|
ctx.body = await userSdk.invite(ctx.request.body)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const removeMultipleInvites = async (
|
||||||
|
ctx: Ctx<DeleteInviteUsersRequest>
|
||||||
|
) => {
|
||||||
|
const inviteCodesToRemove = ctx.request.body.map(
|
||||||
|
(invite: DeleteInviteUserRequest) => invite.code
|
||||||
|
)
|
||||||
|
for (const code of inviteCodesToRemove) {
|
||||||
|
await cache.invite.deleteCode(code)
|
||||||
|
}
|
||||||
|
ctx.body = {
|
||||||
|
message: "User invites successfully removed.",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export const checkInvite = async (ctx: any) => {
|
export const checkInvite = async (ctx: any) => {
|
||||||
const { code } = ctx.params
|
const { code } = ctx.params
|
||||||
let invite
|
let invite
|
||||||
|
|
|
@ -108,6 +108,11 @@ router
|
||||||
buildInviteMultipleValidation(),
|
buildInviteMultipleValidation(),
|
||||||
controller.inviteMultiple
|
controller.inviteMultiple
|
||||||
)
|
)
|
||||||
|
.post(
|
||||||
|
"/api/global/users/multi/invite/delete",
|
||||||
|
auth.builderOrAdmin,
|
||||||
|
controller.removeMultipleInvites
|
||||||
|
)
|
||||||
|
|
||||||
// non-global endpoints
|
// non-global endpoints
|
||||||
.get("/api/global/users/invite/:code", controller.checkInvite)
|
.get("/api/global/users/invite/:code", controller.checkInvite)
|
||||||
|
|
67
yarn.lock
67
yarn.lock
|
@ -6348,6 +6348,11 @@
|
||||||
dependencies:
|
dependencies:
|
||||||
"@types/estree" "*"
|
"@types/estree" "*"
|
||||||
|
|
||||||
|
"@types/tmp@0.2.6":
|
||||||
|
version "0.2.6"
|
||||||
|
resolved "https://registry.yarnpkg.com/@types/tmp/-/tmp-0.2.6.tgz#d785ee90c52d7cc020e249c948c36f7b32d1e217"
|
||||||
|
integrity sha512-chhaNf2oKHlRkDGt+tiKE2Z5aJ6qalm7Z9rlLdBwmOiAAf09YQvvoLXjWK4HWPF1xU/fqvMgfNfpVoBscA/tKA==
|
||||||
|
|
||||||
"@types/tough-cookie@*", "@types/tough-cookie@^4.0.2":
|
"@types/tough-cookie@*", "@types/tough-cookie@^4.0.2":
|
||||||
version "4.0.2"
|
version "4.0.2"
|
||||||
resolved "https://registry.yarnpkg.com/@types/tough-cookie/-/tough-cookie-4.0.2.tgz#6286b4c7228d58ab7866d19716f3696e03a09397"
|
resolved "https://registry.yarnpkg.com/@types/tough-cookie/-/tough-cookie-4.0.2.tgz#6286b4c7228d58ab7866d19716f3696e03a09397"
|
||||||
|
@ -7700,7 +7705,7 @@ bl@^4.0.3, bl@^4.1.0:
|
||||||
inherits "^2.0.4"
|
inherits "^2.0.4"
|
||||||
readable-stream "^3.4.0"
|
readable-stream "^3.4.0"
|
||||||
|
|
||||||
bl@^6.0.12, bl@^6.0.3:
|
bl@^6.0.3:
|
||||||
version "6.0.12"
|
version "6.0.12"
|
||||||
resolved "https://registry.yarnpkg.com/bl/-/bl-6.0.12.tgz#77c35b96e13aeff028496c798b75389ddee9c7f8"
|
resolved "https://registry.yarnpkg.com/bl/-/bl-6.0.12.tgz#77c35b96e13aeff028496c798b75389ddee9c7f8"
|
||||||
integrity sha512-EnEYHilP93oaOa2MnmNEjAcovPS3JlQZOyzGXi3EyEpPhm9qWvdDp7BmAVEVusGzp8LlwQK56Av+OkDoRjzE0w==
|
integrity sha512-EnEYHilP93oaOa2MnmNEjAcovPS3JlQZOyzGXi3EyEpPhm9qWvdDp7BmAVEVusGzp8LlwQK56Av+OkDoRjzE0w==
|
||||||
|
@ -11904,6 +11909,17 @@ glob@^10.0.0, glob@^10.2.2:
|
||||||
minipass "^7.0.4"
|
minipass "^7.0.4"
|
||||||
path-scurry "^1.10.2"
|
path-scurry "^1.10.2"
|
||||||
|
|
||||||
|
glob@^10.3.7:
|
||||||
|
version "10.4.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/glob/-/glob-10.4.1.tgz#0cfb01ab6a6b438177bfe6a58e2576f6efe909c2"
|
||||||
|
integrity sha512-2jelhlq3E4ho74ZyVLN03oKdAZVUa6UDZzFLVH1H7dnoax+y9qyaq8zBkfDIggjniU19z0wU18y16jMB2eyVIw==
|
||||||
|
dependencies:
|
||||||
|
foreground-child "^3.1.0"
|
||||||
|
jackspeak "^3.1.2"
|
||||||
|
minimatch "^9.0.4"
|
||||||
|
minipass "^7.1.2"
|
||||||
|
path-scurry "^1.11.1"
|
||||||
|
|
||||||
glob@^5.0.15:
|
glob@^5.0.15:
|
||||||
version "5.0.15"
|
version "5.0.15"
|
||||||
resolved "https://registry.yarnpkg.com/glob/-/glob-5.0.15.tgz#1bc936b9e02f4a603fcc222ecf7633d30b8b93b1"
|
resolved "https://registry.yarnpkg.com/glob/-/glob-5.0.15.tgz#1bc936b9e02f4a603fcc222ecf7633d30b8b93b1"
|
||||||
|
@ -13472,6 +13488,15 @@ jackspeak@^2.3.6:
|
||||||
optionalDependencies:
|
optionalDependencies:
|
||||||
"@pkgjs/parseargs" "^0.11.0"
|
"@pkgjs/parseargs" "^0.11.0"
|
||||||
|
|
||||||
|
jackspeak@^3.1.2:
|
||||||
|
version "3.1.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/jackspeak/-/jackspeak-3.1.2.tgz#eada67ea949c6b71de50f1b09c92a961897b90ab"
|
||||||
|
integrity sha512-kWmLKn2tRtfYMF/BakihVVRzBKOxz4gJMiL2Rj91WnAB5TPZumSH99R/Yf1qE1u4uRimvCSJfm6hnxohXeEXjQ==
|
||||||
|
dependencies:
|
||||||
|
"@isaacs/cliui" "^8.0.2"
|
||||||
|
optionalDependencies:
|
||||||
|
"@pkgjs/parseargs" "^0.11.0"
|
||||||
|
|
||||||
jake@^10.8.5:
|
jake@^10.8.5:
|
||||||
version "10.8.5"
|
version "10.8.5"
|
||||||
resolved "https://registry.yarnpkg.com/jake/-/jake-10.8.5.tgz#f2183d2c59382cb274226034543b9c03b8164c46"
|
resolved "https://registry.yarnpkg.com/jake/-/jake-10.8.5.tgz#f2183d2c59382cb274226034543b9c03b8164c46"
|
||||||
|
@ -15751,6 +15776,13 @@ minimatch@^8.0.2:
|
||||||
dependencies:
|
dependencies:
|
||||||
brace-expansion "^2.0.1"
|
brace-expansion "^2.0.1"
|
||||||
|
|
||||||
|
minimatch@^9.0.4:
|
||||||
|
version "9.0.4"
|
||||||
|
resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-9.0.4.tgz#8e49c731d1749cbec05050ee5145147b32496a51"
|
||||||
|
integrity sha512-KqWh+VchfxcMNRAJjj2tnsSJdNbHsVgnkBhTNrW7AjVo6OvLtxw8zfT9oLw1JSohlFzJ8jCoTgaoXvJ+kHt6fw==
|
||||||
|
dependencies:
|
||||||
|
brace-expansion "^2.0.1"
|
||||||
|
|
||||||
minimist-options@4.1.0:
|
minimist-options@4.1.0:
|
||||||
version "4.1.0"
|
version "4.1.0"
|
||||||
resolved "https://registry.yarnpkg.com/minimist-options/-/minimist-options-4.1.0.tgz#c0655713c53a8a2ebd77ffa247d342c40f010619"
|
resolved "https://registry.yarnpkg.com/minimist-options/-/minimist-options-4.1.0.tgz#c0655713c53a8a2ebd77ffa247d342c40f010619"
|
||||||
|
@ -15845,6 +15877,11 @@ minipass@^5.0.0:
|
||||||
resolved "https://registry.yarnpkg.com/minipass/-/minipass-7.0.4.tgz#dbce03740f50a4786ba994c1fb908844d27b038c"
|
resolved "https://registry.yarnpkg.com/minipass/-/minipass-7.0.4.tgz#dbce03740f50a4786ba994c1fb908844d27b038c"
|
||||||
integrity sha512-jYofLM5Dam9279rdkWzqHozUo4ybjdZmCsDHePy5V/PbBcVMiSZR97gmAy45aqi8CK1lG2ECd356FU86avfwUQ==
|
integrity sha512-jYofLM5Dam9279rdkWzqHozUo4ybjdZmCsDHePy5V/PbBcVMiSZR97gmAy45aqi8CK1lG2ECd356FU86avfwUQ==
|
||||||
|
|
||||||
|
minipass@^7.1.2:
|
||||||
|
version "7.1.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/minipass/-/minipass-7.1.2.tgz#93a9626ce5e5e66bd4db86849e7515e92340a707"
|
||||||
|
integrity sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==
|
||||||
|
|
||||||
minizlib@^2.1.1, minizlib@^2.1.2:
|
minizlib@^2.1.1, minizlib@^2.1.2:
|
||||||
version "2.1.2"
|
version "2.1.2"
|
||||||
resolved "https://registry.yarnpkg.com/minizlib/-/minizlib-2.1.2.tgz#e90d3466ba209b932451508a11ce3d3632145931"
|
resolved "https://registry.yarnpkg.com/minizlib/-/minizlib-2.1.2.tgz#e90d3466ba209b932451508a11ce3d3632145931"
|
||||||
|
@ -16033,10 +16070,10 @@ mute-stream@~1.0.0:
|
||||||
resolved "https://registry.yarnpkg.com/mute-stream/-/mute-stream-1.0.0.tgz#e31bd9fe62f0aed23520aa4324ea6671531e013e"
|
resolved "https://registry.yarnpkg.com/mute-stream/-/mute-stream-1.0.0.tgz#e31bd9fe62f0aed23520aa4324ea6671531e013e"
|
||||||
integrity sha512-avsJQhyd+680gKXyG/sQc0nXaC6rBkPOfyHYcFb9+hdkqQkR9bdnkJ0AMZhke0oesPqIO+mFFJ+IdBc7mst4IA==
|
integrity sha512-avsJQhyd+680gKXyG/sQc0nXaC6rBkPOfyHYcFb9+hdkqQkR9bdnkJ0AMZhke0oesPqIO+mFFJ+IdBc7mst4IA==
|
||||||
|
|
||||||
mysql2@3.9.7:
|
mysql2@3.9.8:
|
||||||
version "3.9.7"
|
version "3.9.8"
|
||||||
resolved "https://registry.yarnpkg.com/mysql2/-/mysql2-3.9.7.tgz#843755daf65b5ef08afe545fe14b8fb62824741a"
|
resolved "https://registry.yarnpkg.com/mysql2/-/mysql2-3.9.8.tgz#fe8a0f975f2c495ed76ca988ddc5505801dc49ce"
|
||||||
integrity sha512-KnJT8vYRcNAZv73uf9zpXqNbvBG7DJrs+1nACsjZP1HMJ1TgXEy8wnNilXAn/5i57JizXKtrUtwDB7HxT9DDpw==
|
integrity sha512-+5JKNjPuks1FNMoy9TYpl77f+5frbTklz7eb3XDwbpsERRLEeXiW2PDEkakYF50UuKU2qwfGnyXpKYvukv8mGA==
|
||||||
dependencies:
|
dependencies:
|
||||||
denque "^2.1.0"
|
denque "^2.1.0"
|
||||||
generate-function "^2.3.1"
|
generate-function "^2.3.1"
|
||||||
|
@ -17378,6 +17415,14 @@ path-scurry@^1.10.2, path-scurry@^1.6.1:
|
||||||
lru-cache "^10.2.0"
|
lru-cache "^10.2.0"
|
||||||
minipass "^5.0.0 || ^6.0.2 || ^7.0.0"
|
minipass "^5.0.0 || ^6.0.2 || ^7.0.0"
|
||||||
|
|
||||||
|
path-scurry@^1.11.1:
|
||||||
|
version "1.11.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/path-scurry/-/path-scurry-1.11.1.tgz#7960a668888594a0720b12a911d1a742ab9f11d2"
|
||||||
|
integrity sha512-Xa4Nw17FS9ApQFJ9umLiJS4orGjm7ZzwUrwamcGQuHSzDyth9boKDaycYdDcZDuqYATXw4HFXgaqWTctW/v1HA==
|
||||||
|
dependencies:
|
||||||
|
lru-cache "^10.2.0"
|
||||||
|
minipass "^5.0.0 || ^6.0.2 || ^7.0.0"
|
||||||
|
|
||||||
path-to-regexp@1.x:
|
path-to-regexp@1.x:
|
||||||
version "1.8.0"
|
version "1.8.0"
|
||||||
resolved "https://registry.yarnpkg.com/path-to-regexp/-/path-to-regexp-1.8.0.tgz#887b3ba9d84393e87a0a0b9f4cb756198b53548a"
|
resolved "https://registry.yarnpkg.com/path-to-regexp/-/path-to-regexp-1.8.0.tgz#887b3ba9d84393e87a0a0b9f4cb756198b53548a"
|
||||||
|
@ -19318,6 +19363,13 @@ rimraf@^4.4.1:
|
||||||
dependencies:
|
dependencies:
|
||||||
glob "^9.2.0"
|
glob "^9.2.0"
|
||||||
|
|
||||||
|
rimraf@^5.0.7:
|
||||||
|
version "5.0.7"
|
||||||
|
resolved "https://registry.yarnpkg.com/rimraf/-/rimraf-5.0.7.tgz#27bddf202e7d89cb2e0381656380d1734a854a74"
|
||||||
|
integrity sha512-nV6YcJo5wbLW77m+8KjH8aB/7/rxQy9SZ0HY5shnwULfS+9nmTtVXAJET5NdZmCzA4fPI/Hm1wo/Po/4mopOdg==
|
||||||
|
dependencies:
|
||||||
|
glob "^10.3.7"
|
||||||
|
|
||||||
ripemd160@^2.0.0, ripemd160@^2.0.1:
|
ripemd160@^2.0.0, ripemd160@^2.0.1:
|
||||||
version "2.0.2"
|
version "2.0.2"
|
||||||
resolved "https://registry.yarnpkg.com/ripemd160/-/ripemd160-2.0.2.tgz#a1c1a6f624751577ba5d07914cbc92850585890c"
|
resolved "https://registry.yarnpkg.com/ripemd160/-/ripemd160-2.0.2.tgz#a1c1a6f624751577ba5d07914cbc92850585890c"
|
||||||
|
@ -21236,6 +21288,11 @@ tlhunter-sorted-set@^0.1.0:
|
||||||
resolved "https://registry.yarnpkg.com/tlhunter-sorted-set/-/tlhunter-sorted-set-0.1.0.tgz#1c3eae28c0fa4dff97e9501d2e3c204b86406f4b"
|
resolved "https://registry.yarnpkg.com/tlhunter-sorted-set/-/tlhunter-sorted-set-0.1.0.tgz#1c3eae28c0fa4dff97e9501d2e3c204b86406f4b"
|
||||||
integrity sha512-eGYW4bjf1DtrHzUYxYfAcSytpOkA44zsr7G2n3PV7yOUR23vmkGe3LL4R+1jL9OsXtbsFOwe8XtbCrabeaEFnw==
|
integrity sha512-eGYW4bjf1DtrHzUYxYfAcSytpOkA44zsr7G2n3PV7yOUR23vmkGe3LL4R+1jL9OsXtbsFOwe8XtbCrabeaEFnw==
|
||||||
|
|
||||||
|
tmp@0.2.3:
|
||||||
|
version "0.2.3"
|
||||||
|
resolved "https://registry.yarnpkg.com/tmp/-/tmp-0.2.3.tgz#eb783cc22bc1e8bebd0671476d46ea4eb32a79ae"
|
||||||
|
integrity sha512-nZD7m9iCPC5g0pYmcaxogYKggSfLsdxl8of3Q/oIbqCqLLIO9IAF0GWjX1z9NZRHPiXv8Wex4yDCaZsgEw0Y8w==
|
||||||
|
|
||||||
tmp@^0.0.33:
|
tmp@^0.0.33:
|
||||||
version "0.0.33"
|
version "0.0.33"
|
||||||
resolved "https://registry.yarnpkg.com/tmp/-/tmp-0.0.33.tgz#6d34335889768d21b2bcda0aa277ced3b1bfadf9"
|
resolved "https://registry.yarnpkg.com/tmp/-/tmp-0.0.33.tgz#6d34335889768d21b2bcda0aa277ced3b1bfadf9"
|
||||||
|
|
Loading…
Reference in New Issue