🐛 (dify) Fix Dify error when inputs are empty
This commit is contained in:
@@ -23,6 +23,7 @@ import { ForgedBlockDefinition, ForgedBlock } from '@typebot.io/forge-schemas'
|
|||||||
import { PrimitiveList } from '@/components/PrimitiveList'
|
import { PrimitiveList } from '@/components/PrimitiveList'
|
||||||
import { SwitchWithLabel } from '@/components/inputs/SwitchWithLabel'
|
import { SwitchWithLabel } from '@/components/inputs/SwitchWithLabel'
|
||||||
import { CodeEditor } from '@/components/inputs/CodeEditor'
|
import { CodeEditor } from '@/components/inputs/CodeEditor'
|
||||||
|
import { getZodInnerSchema } from '../../helpers/getZodInnerSchema'
|
||||||
|
|
||||||
const mdComponents = {
|
const mdComponents = {
|
||||||
a: ({ href, children }) => (
|
a: ({ href, children }) => (
|
||||||
@@ -57,16 +58,14 @@ export const ZodFieldLayout = ({
|
|||||||
propName?: string
|
propName?: string
|
||||||
onDataChange: (val: any) => void
|
onDataChange: (val: any) => void
|
||||||
}) => {
|
}) => {
|
||||||
const layout = schema._def.layout as ZodLayoutMetadata<ZodTypeAny> | undefined
|
const innerSchema = getZodInnerSchema(schema)
|
||||||
const type = schema._def.innerType
|
const layout = innerSchema._def.layout
|
||||||
? schema._def.innerType._def.typeName
|
|
||||||
: schema._def.typeName
|
|
||||||
|
|
||||||
switch (type) {
|
switch (innerSchema._def.typeName) {
|
||||||
case 'ZodObject':
|
case 'ZodObject':
|
||||||
return (
|
return (
|
||||||
<ZodObjectLayout
|
<ZodObjectLayout
|
||||||
schema={schema as z.ZodObject<any>}
|
schema={innerSchema as z.ZodObject<any>}
|
||||||
data={data}
|
data={data}
|
||||||
onDataChange={onDataChange}
|
onDataChange={onDataChange}
|
||||||
isInAccordion={isInAccordion}
|
isInAccordion={isInAccordion}
|
||||||
@@ -77,10 +76,12 @@ export const ZodFieldLayout = ({
|
|||||||
case 'ZodDiscriminatedUnion': {
|
case 'ZodDiscriminatedUnion': {
|
||||||
return (
|
return (
|
||||||
<ZodDiscriminatedUnionLayout
|
<ZodDiscriminatedUnionLayout
|
||||||
discriminant={schema._def.discriminator}
|
discriminant={innerSchema._def.discriminator}
|
||||||
data={data}
|
data={data}
|
||||||
schema={schema as z.ZodDiscriminatedUnion<string, z.ZodObject<any>[]>}
|
schema={
|
||||||
dropdownPlaceholder={`Select a ${schema._def.discriminator}`}
|
innerSchema as z.ZodDiscriminatedUnion<string, z.ZodObject<any>[]>
|
||||||
|
}
|
||||||
|
dropdownPlaceholder={`Select a ${innerSchema._def.discriminator}`}
|
||||||
onDataChange={onDataChange}
|
onDataChange={onDataChange}
|
||||||
/>
|
/>
|
||||||
)
|
)
|
||||||
@@ -99,7 +100,7 @@ export const ZodFieldLayout = ({
|
|||||||
<AccordionPanel as={Stack} pt="4">
|
<AccordionPanel as={Stack} pt="4">
|
||||||
<ZodArrayContent
|
<ZodArrayContent
|
||||||
data={data}
|
data={data}
|
||||||
schema={schema}
|
schema={innerSchema}
|
||||||
blockDef={blockDef}
|
blockDef={blockDef}
|
||||||
blockOptions={blockOptions}
|
blockOptions={blockOptions}
|
||||||
layout={layout}
|
layout={layout}
|
||||||
@@ -113,7 +114,7 @@ export const ZodFieldLayout = ({
|
|||||||
return (
|
return (
|
||||||
<ZodArrayContent
|
<ZodArrayContent
|
||||||
data={data}
|
data={data}
|
||||||
schema={schema}
|
schema={innerSchema}
|
||||||
blockDef={blockDef}
|
blockDef={blockDef}
|
||||||
blockOptions={blockOptions}
|
blockOptions={blockOptions}
|
||||||
layout={layout}
|
layout={layout}
|
||||||
@@ -126,7 +127,7 @@ export const ZodFieldLayout = ({
|
|||||||
<DropdownList
|
<DropdownList
|
||||||
currentItem={data ?? layout?.defaultValue}
|
currentItem={data ?? layout?.defaultValue}
|
||||||
onItemSelect={onDataChange}
|
onItemSelect={onDataChange}
|
||||||
items={schema._def.innerType._def.values}
|
items={innerSchema._def.values}
|
||||||
label={layout?.label}
|
label={layout?.label}
|
||||||
helperText={
|
helperText={
|
||||||
layout?.helperText ? (
|
layout?.helperText ? (
|
||||||
@@ -295,7 +296,7 @@ const ZodArrayContent = ({
|
|||||||
isInAccordion?: boolean
|
isInAccordion?: boolean
|
||||||
onDataChange: (val: any) => void
|
onDataChange: (val: any) => void
|
||||||
}) => {
|
}) => {
|
||||||
const type = schema._def.innerType._def.type._def.innerType?._def.typeName
|
const type = schema._def.type._def.innerType?._def.typeName
|
||||||
if (type === 'ZodString' || type === 'ZodNumber' || type === 'ZodEnum')
|
if (type === 'ZodString' || type === 'ZodNumber' || type === 'ZodEnum')
|
||||||
return (
|
return (
|
||||||
<Stack spacing={0}>
|
<Stack spacing={0}>
|
||||||
@@ -310,7 +311,7 @@ const ZodArrayContent = ({
|
|||||||
>
|
>
|
||||||
{({ item, onItemChange }) => (
|
{({ item, onItemChange }) => (
|
||||||
<ZodFieldLayout
|
<ZodFieldLayout
|
||||||
schema={schema._def.innerType._def.type}
|
schema={schema._def.type}
|
||||||
data={item}
|
data={item}
|
||||||
blockDef={blockDef}
|
blockDef={blockDef}
|
||||||
blockOptions={blockOptions}
|
blockOptions={blockOptions}
|
||||||
@@ -335,7 +336,7 @@ const ZodArrayContent = ({
|
|||||||
{({ item, onItemChange }) => (
|
{({ item, onItemChange }) => (
|
||||||
<Stack p="4" rounded="md" flex="1" borderWidth="1px" maxW="100%">
|
<Stack p="4" rounded="md" flex="1" borderWidth="1px" maxW="100%">
|
||||||
<ZodFieldLayout
|
<ZodFieldLayout
|
||||||
schema={schema._def.innerType._def.type}
|
schema={schema._def.type}
|
||||||
blockDef={blockDef}
|
blockDef={blockDef}
|
||||||
blockOptions={blockOptions}
|
blockOptions={blockOptions}
|
||||||
data={item}
|
data={item}
|
||||||
|
|||||||
@@ -14,6 +14,7 @@ import { ReactNode } from 'react'
|
|||||||
import { ZodTypeAny } from 'zod'
|
import { ZodTypeAny } from 'zod'
|
||||||
import { ZodFieldLayout } from './ZodFieldLayout'
|
import { ZodFieldLayout } from './ZodFieldLayout'
|
||||||
import { ForgedBlockDefinition, ForgedBlock } from '@typebot.io/forge-schemas'
|
import { ForgedBlockDefinition, ForgedBlock } from '@typebot.io/forge-schemas'
|
||||||
|
import { getZodInnerSchema } from '../../helpers/getZodInnerSchema'
|
||||||
|
|
||||||
export const ZodObjectLayout = ({
|
export const ZodObjectLayout = ({
|
||||||
schema,
|
schema,
|
||||||
@@ -38,7 +39,7 @@ export const ZodObjectLayout = ({
|
|||||||
}>(
|
}>(
|
||||||
(nodes, key, index) => {
|
(nodes, key, index) => {
|
||||||
if (ignoreKeys?.includes(key)) return nodes
|
if (ignoreKeys?.includes(key)) return nodes
|
||||||
const keySchema = schema.shape[key]
|
const keySchema = getZodInnerSchema(schema.shape[key])
|
||||||
const layout = keySchema._def.layout as
|
const layout = keySchema._def.layout as
|
||||||
| ZodLayoutMetadata<ZodTypeAny>
|
| ZodLayoutMetadata<ZodTypeAny>
|
||||||
| undefined
|
| undefined
|
||||||
@@ -46,7 +47,7 @@ export const ZodObjectLayout = ({
|
|||||||
layout &&
|
layout &&
|
||||||
layout.accordion &&
|
layout.accordion &&
|
||||||
!isInAccordion &&
|
!isInAccordion &&
|
||||||
keySchema._def.innerType._def.typeName !== 'ZodArray'
|
keySchema._def.typeName !== 'ZodArray'
|
||||||
) {
|
) {
|
||||||
if (nodes.accordionsCreated.includes(layout.accordion)) return nodes
|
if (nodes.accordionsCreated.includes(layout.accordion)) return nodes
|
||||||
const accordionKeys = getObjectKeysWithSameAccordionAttr(
|
const accordionKeys = getObjectKeysWithSameAccordionAttr(
|
||||||
|
|||||||
18
apps/builder/src/features/forge/helpers/getZodInnerSchema.ts
Normal file
18
apps/builder/src/features/forge/helpers/getZodInnerSchema.ts
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
import { z } from '@typebot.io/forge/zod'
|
||||||
|
|
||||||
|
export const getZodInnerSchema = (schema: z.ZodTypeAny): z.ZodTypeAny => {
|
||||||
|
if (schema._def.typeName === 'ZodEffects')
|
||||||
|
return getZodInnerSchema(schema._def.schema)
|
||||||
|
if (schema._def.typeName === 'ZodOptional') {
|
||||||
|
const innerSchema = getZodInnerSchema(schema._def.innerType)
|
||||||
|
return {
|
||||||
|
...innerSchema,
|
||||||
|
_def: {
|
||||||
|
...innerSchema._def,
|
||||||
|
layout: schema._def.layout,
|
||||||
|
},
|
||||||
|
} as z.ZodTypeAny
|
||||||
|
}
|
||||||
|
|
||||||
|
return schema
|
||||||
|
}
|
||||||
@@ -20,6 +20,7 @@ WhatsApp environment have some limitations that you need to keep in mind when bu
|
|||||||
|
|
||||||
- GIF and SVG image files are not supported. They won't be displayed.
|
- GIF and SVG image files are not supported. They won't be displayed.
|
||||||
- Buttons content can't be longer than 20 characters. If the content is longer, it will be truncated.
|
- Buttons content can't be longer than 20 characters. If the content is longer, it will be truncated.
|
||||||
|
- WhatsApp only allows to display 3 buttons at a time. So we work around that by adding "..." messages to display more buttons.
|
||||||
- Incompatible blocks, if present, they will be skipped:
|
- Incompatible blocks, if present, they will be skipped:
|
||||||
|
|
||||||
- Payment input block
|
- Payment input block
|
||||||
|
|||||||
@@ -1,4 +0,0 @@
|
|||||||
const urlRegex =
|
|
||||||
/^https?:\/\/(www\.)?[-a-zA-Z0-9@:%._\+~#=]{1,256}\.[a-zA-Z0-9()]{1,6}\b([-a-zA-Z0-9()@:%_\+.~#?&//=]*)$/
|
|
||||||
|
|
||||||
export const validateUrl = (url: string) => urlRegex.test(url)
|
|
||||||
@@ -11,7 +11,6 @@ import { executeGroup, parseInput } from './executeGroup'
|
|||||||
import { getNextGroup } from './getNextGroup'
|
import { getNextGroup } from './getNextGroup'
|
||||||
import { validateEmail } from './blocks/inputs/email/validateEmail'
|
import { validateEmail } from './blocks/inputs/email/validateEmail'
|
||||||
import { formatPhoneNumber } from './blocks/inputs/phone/formatPhoneNumber'
|
import { formatPhoneNumber } from './blocks/inputs/phone/formatPhoneNumber'
|
||||||
import { validateUrl } from './blocks/inputs/url/validateUrl'
|
|
||||||
import { resumeWebhookExecution } from './blocks/integrations/webhook/resumeWebhookExecution'
|
import { resumeWebhookExecution } from './blocks/integrations/webhook/resumeWebhookExecution'
|
||||||
import { upsertAnswer } from './queries/upsertAnswer'
|
import { upsertAnswer } from './queries/upsertAnswer'
|
||||||
import { parseButtonsReply } from './blocks/inputs/buttons/parseButtonsReply'
|
import { parseButtonsReply } from './blocks/inputs/buttons/parseButtonsReply'
|
||||||
@@ -42,6 +41,7 @@ import { resumeChatCompletion } from './blocks/integrations/legacy/openai/resume
|
|||||||
import { env } from '@typebot.io/env'
|
import { env } from '@typebot.io/env'
|
||||||
import { downloadMedia } from './whatsapp/downloadMedia'
|
import { downloadMedia } from './whatsapp/downloadMedia'
|
||||||
import { uploadFileToBucket } from '@typebot.io/lib/s3/uploadFileToBucket'
|
import { uploadFileToBucket } from '@typebot.io/lib/s3/uploadFileToBucket'
|
||||||
|
import { isURL } from '@typebot.io/lib/validators/isURL'
|
||||||
|
|
||||||
type Params = {
|
type Params = {
|
||||||
version: 1 | 2
|
version: 1 | 2
|
||||||
@@ -450,7 +450,7 @@ const parseReply =
|
|||||||
}
|
}
|
||||||
case InputBlockType.URL: {
|
case InputBlockType.URL: {
|
||||||
if (!reply) return { status: 'fail' }
|
if (!reply) return { status: 'fail' }
|
||||||
const isValid = validateUrl(reply)
|
const isValid = isURL(reply, { require_protocol: false })
|
||||||
if (!isValid) return { status: 'fail' }
|
if (!isValid) return { status: 'fail' }
|
||||||
return { status: 'success', reply: reply }
|
return { status: 'success', reply: reply }
|
||||||
}
|
}
|
||||||
@@ -477,7 +477,7 @@ const parseReply =
|
|||||||
? { status: 'fail' }
|
? { status: 'fail' }
|
||||||
: { status: 'skip' }
|
: { status: 'skip' }
|
||||||
const urls = reply.split(', ')
|
const urls = reply.split(', ')
|
||||||
const status = urls.some((url) => validateUrl(url)) ? 'success' : 'fail'
|
const status = urls.some((url) => isURL(url)) ? 'success' : 'fail'
|
||||||
return { status, reply: reply }
|
return { status, reply: reply }
|
||||||
}
|
}
|
||||||
case InputBlockType.PAYMENT: {
|
case InputBlockType.PAYMENT: {
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { createAction, option } from '@typebot.io/forge'
|
import { createAction, option } from '@typebot.io/forge'
|
||||||
import { isDefined, isEmpty } from '@typebot.io/lib'
|
import { isDefined, isEmpty } from '@typebot.io/lib'
|
||||||
import { got } from 'got'
|
import { HTTPError, got } from 'got'
|
||||||
import { auth } from '../auth'
|
import { auth } from '../auth'
|
||||||
import { DifyResponse } from '../types'
|
import { DifyResponse } from '../types'
|
||||||
import { defaultBaseUrl } from '../constants'
|
import { defaultBaseUrl } from '../constants'
|
||||||
@@ -41,25 +41,28 @@ export const createChatMessage = createAction({
|
|||||||
credentials: { apiEndpoint, apiKey },
|
credentials: { apiEndpoint, apiKey },
|
||||||
options: { conversation_id, query, user, inputs, responseMapping },
|
options: { conversation_id, query, user, inputs, responseMapping },
|
||||||
variables,
|
variables,
|
||||||
|
logs,
|
||||||
}) => {
|
}) => {
|
||||||
|
try {
|
||||||
const res: DifyResponse = await got
|
const res: DifyResponse = await got
|
||||||
.post((apiEndpoint ?? defaultBaseUrl) + '/v1/chat-messages', {
|
.post((apiEndpoint ?? defaultBaseUrl) + '/v1/chat-messages', {
|
||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${apiKey}`,
|
Authorization: `Bearer ${apiKey}`,
|
||||||
},
|
},
|
||||||
json: {
|
json: {
|
||||||
inputs: inputs?.reduce((acc, { key, value }) => {
|
inputs:
|
||||||
|
inputs?.reduce((acc, { key, value }) => {
|
||||||
if (isEmpty(key) || isEmpty(value)) return acc
|
if (isEmpty(key) || isEmpty(value)) return acc
|
||||||
return {
|
return {
|
||||||
...acc,
|
...acc,
|
||||||
[key]: value,
|
[key]: value,
|
||||||
}
|
}
|
||||||
}, {}),
|
}, {}) ?? {},
|
||||||
query,
|
query,
|
||||||
response_mode: 'blocking',
|
response_mode: 'blocking',
|
||||||
conversation_id,
|
conversation_id,
|
||||||
user,
|
user,
|
||||||
files: []
|
files: [],
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
.json()
|
.json()
|
||||||
@@ -76,6 +79,15 @@ export const createChatMessage = createAction({
|
|||||||
if (item === 'Total Tokens')
|
if (item === 'Total Tokens')
|
||||||
variables.set(mapping.variableId, res.metadata.usage.total_tokens)
|
variables.set(mapping.variableId, res.metadata.usage.total_tokens)
|
||||||
})
|
})
|
||||||
|
} catch (error) {
|
||||||
|
if (error instanceof HTTPError)
|
||||||
|
return logs.add({
|
||||||
|
status: 'error',
|
||||||
|
description: error.message,
|
||||||
|
details: error.response.body,
|
||||||
|
})
|
||||||
|
console.error(error)
|
||||||
|
}
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -1,17 +1,27 @@
|
|||||||
import { option, AuthDefinition } from '@typebot.io/forge'
|
import { option, AuthDefinition } from '@typebot.io/forge'
|
||||||
import { defaultBaseUrl } from './constants'
|
import { defaultBaseUrl } from './constants'
|
||||||
|
import { isURL } from '@typebot.io/lib/validators/isURL'
|
||||||
|
|
||||||
|
const extractBaseUrl = (val: string | undefined) => {
|
||||||
|
if (!val) return val
|
||||||
|
const url = new URL(val)
|
||||||
|
return url.origin
|
||||||
|
}
|
||||||
|
|
||||||
export const auth = {
|
export const auth = {
|
||||||
type: 'encryptedCredentials',
|
type: 'encryptedCredentials',
|
||||||
name: 'Dify.AI account',
|
name: 'Dify.AI account',
|
||||||
schema: option.object({
|
schema: option.object({
|
||||||
apiEndpoint: option.string.layout({
|
apiEndpoint: option.string
|
||||||
|
.layout({
|
||||||
label: 'API Endpoint',
|
label: 'API Endpoint',
|
||||||
isRequired: true,
|
isRequired: true,
|
||||||
helperText: 'URI where the Service API is hosted.',
|
helperText: 'URI where the Service API is hosted.',
|
||||||
withVariableButton: false,
|
withVariableButton: false,
|
||||||
defaultValue: defaultBaseUrl,
|
defaultValue: defaultBaseUrl,
|
||||||
}),
|
})
|
||||||
|
.refine((val) => !val || isURL(val))
|
||||||
|
.transform(extractBaseUrl),
|
||||||
apiKey: option.string.layout({
|
apiKey: option.string.layout({
|
||||||
label: 'App API key',
|
label: 'App API key',
|
||||||
isRequired: true,
|
isRequired: true,
|
||||||
|
|||||||
@@ -15,6 +15,7 @@
|
|||||||
"@typebot.io/tsconfig": "workspace:*",
|
"@typebot.io/tsconfig": "workspace:*",
|
||||||
"@types/escape-html": "^1.0.4",
|
"@types/escape-html": "^1.0.4",
|
||||||
"@types/nodemailer": "6.4.8",
|
"@types/nodemailer": "6.4.8",
|
||||||
|
"@types/validator": "13.11.9",
|
||||||
"next": "14.1.0",
|
"next": "14.1.0",
|
||||||
"nodemailer": "6.9.3",
|
"nodemailer": "6.9.3",
|
||||||
"typescript": "5.3.2"
|
"typescript": "5.3.2"
|
||||||
@@ -44,7 +45,7 @@
|
|||||||
"remark-parse": "11.0.0",
|
"remark-parse": "11.0.0",
|
||||||
"stripe": "12.13.0",
|
"stripe": "12.13.0",
|
||||||
"unified": "11.0.4",
|
"unified": "11.0.4",
|
||||||
"zod": "3.22.4",
|
"validator": "13.11.0",
|
||||||
"ky": "1.1.3"
|
"zod": "3.22.4"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
10
packages/lib/validators/isURL.ts
Normal file
10
packages/lib/validators/isURL.ts
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
import isURL, { IsURLOptions } from 'validator/lib/isURL'
|
||||||
|
|
||||||
|
const customIsURL = (val: string, options?: IsURLOptions) =>
|
||||||
|
isURL(val, {
|
||||||
|
protocols: ['https', 'http'],
|
||||||
|
require_protocol: true,
|
||||||
|
...options,
|
||||||
|
})
|
||||||
|
|
||||||
|
export { customIsURL as isURL }
|
||||||
15
pnpm-lock.yaml
generated
15
pnpm-lock.yaml
generated
@@ -1479,6 +1479,9 @@ importers:
|
|||||||
unified:
|
unified:
|
||||||
specifier: 11.0.4
|
specifier: 11.0.4
|
||||||
version: 11.0.4
|
version: 11.0.4
|
||||||
|
validator:
|
||||||
|
specifier: ^13.11.0
|
||||||
|
version: 13.11.0
|
||||||
zod:
|
zod:
|
||||||
specifier: 3.22.4
|
specifier: 3.22.4
|
||||||
version: 3.22.4
|
version: 3.22.4
|
||||||
@@ -1510,6 +1513,9 @@ importers:
|
|||||||
'@types/nodemailer':
|
'@types/nodemailer':
|
||||||
specifier: 6.4.8
|
specifier: 6.4.8
|
||||||
version: 6.4.8
|
version: 6.4.8
|
||||||
|
'@types/validator':
|
||||||
|
specifier: ^13.11.9
|
||||||
|
version: 13.11.9
|
||||||
next:
|
next:
|
||||||
specifier: 14.1.0
|
specifier: 14.1.0
|
||||||
version: 14.1.0(@babel/core@7.22.9)(react-dom@18.2.0)(react@18.2.0)
|
version: 14.1.0(@babel/core@7.22.9)(react-dom@18.2.0)(react@18.2.0)
|
||||||
@@ -9661,6 +9667,10 @@ packages:
|
|||||||
/@types/unist@3.0.2:
|
/@types/unist@3.0.2:
|
||||||
resolution: {integrity: sha512-dqId9J8K/vGi5Zr7oo212BGii5m3q5Hxlkwy3WpYuKPklmBEvsbMYYyLxAQpSffdLl/gdW0XUpKWFvYmyoWCoQ==}
|
resolution: {integrity: sha512-dqId9J8K/vGi5Zr7oo212BGii5m3q5Hxlkwy3WpYuKPklmBEvsbMYYyLxAQpSffdLl/gdW0XUpKWFvYmyoWCoQ==}
|
||||||
|
|
||||||
|
/@types/validator@13.11.9:
|
||||||
|
resolution: {integrity: sha512-FCTsikRozryfayPuiI46QzH3fnrOoctTjvOYZkho9BTFLCOZ2rgZJHMOVgCOfttjPJcgOx52EpkY0CMfy87MIw==}
|
||||||
|
dev: true
|
||||||
|
|
||||||
/@types/webpack@5.28.5(@swc/core@1.3.101)(esbuild@0.19.11):
|
/@types/webpack@5.28.5(@swc/core@1.3.101)(esbuild@0.19.11):
|
||||||
resolution: {integrity: sha512-wR87cgvxj3p6D0Crt1r5avwqffqPXUkNlnQ1mjU93G7gCuFjufZR4I6j8cz5g1F1tTYpfOOFvly+cmIQwL9wvw==}
|
resolution: {integrity: sha512-wR87cgvxj3p6D0Crt1r5avwqffqPXUkNlnQ1mjU93G7gCuFjufZR4I6j8cz5g1F1tTYpfOOFvly+cmIQwL9wvw==}
|
||||||
dependencies:
|
dependencies:
|
||||||
@@ -23236,6 +23246,11 @@ packages:
|
|||||||
spdx-correct: 3.2.0
|
spdx-correct: 3.2.0
|
||||||
spdx-expression-parse: 3.0.1
|
spdx-expression-parse: 3.0.1
|
||||||
|
|
||||||
|
/validator@13.11.0:
|
||||||
|
resolution: {integrity: sha512-Ii+sehpSfZy+At5nPdnyMhx78fEoPDkR2XW/zimHEL3MyGJQOCQ7WeP20jPYRz7ZCpcKLB21NxuXHF3bxjStBQ==}
|
||||||
|
engines: {node: '>= 0.10'}
|
||||||
|
dev: false
|
||||||
|
|
||||||
/vary@1.1.2:
|
/vary@1.1.2:
|
||||||
resolution: {integrity: sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==}
|
resolution: {integrity: sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==}
|
||||||
engines: {node: '>= 0.8'}
|
engines: {node: '>= 0.8'}
|
||||||
|
|||||||
Reference in New Issue
Block a user