Merge remote-tracking branch 'upstream/master' into inputdefault

This commit is contained in:
Riccardo Balbo 2024-11-05 16:35:16 +01:00
commit cd4fcdff83
2 changed files with 23 additions and 13 deletions

View File

@ -15,8 +15,8 @@ export const fields = [
validate: string() validate: string()
.matches(/^blink_[A-Za-z0-9]+$/, { message: 'must match pattern blink_A-Za-z0-9' }), .matches(/^blink_[A-Za-z0-9]+$/, { message: 'must match pattern blink_A-Za-z0-9' }),
help: `you can get an API key from [Blink Dashboard](${galoyBlinkDashboardUrl}).\nPlease make sure to select ONLY the 'Read' and 'Write' scopes when generating this API key.`, help: `you can get an API key from [Blink Dashboard](${galoyBlinkDashboardUrl}).\nPlease make sure to select ONLY the 'Read' and 'Write' scopes when generating this API key.`,
optional: 'for sending', requiredWithout: 'apiKeyRecv',
requiredWithout: 'apiKeyRecv' optional: 'for sending'
}, },
{ {
name: 'currency', name: 'currency',
@ -31,33 +31,36 @@ export const fields = [
validate: string() validate: string()
.transform(value => value ? value.toUpperCase() : 'BTC') .transform(value => value ? value.toUpperCase() : 'BTC')
.oneOf(['USD', 'BTC'], 'must be BTC or USD'), .oneOf(['USD', 'BTC'], 'must be BTC or USD'),
optional: 'for sending' optional: 'for sending',
requiredWithout: 'currencyRecv'
}, },
{ {
name: 'apiKeyRecv', name: 'apiKeyRecv',
label: 'api key', label: 'receive api key',
type: 'password', type: 'password',
help: `you can get an API key from [Blink Dashboard](${galoyBlinkDashboardUrl}).\nPlease make sure to select ONLY the 'Read' and 'Receive' scopes when generating this API key.`, help: `you can get an API key from [Blink Dashboard](${galoyBlinkDashboardUrl}).\nPlease make sure to select ONLY the 'Read' and 'Receive' scopes when generating this API key.`,
placeholder: 'blink_...', placeholder: 'blink_...',
optional: 'for receiving',
serverOnly: true, serverOnly: true,
requiredWithout: 'apiKey',
validate: string() validate: string()
.matches(/^blink_[A-Za-z0-9]+$/, { message: 'must match pattern blink_A-Za-z0-9' }) .matches(/^blink_[A-Za-z0-9]+$/, { message: 'must match pattern blink_A-Za-z0-9' }),
optional: 'for receiving',
requiredWithout: 'apiKey'
}, },
{ {
name: 'currencyRecv', name: 'currencyRecv',
label: 'wallet type', label: 'receive wallet type',
type: 'text', type: 'text',
help: 'the blink wallet to use for receiving (only BTC available)', help: 'the blink wallet to use for receiving (only BTC available)',
defaultValue: 'BTC', defaultValue: 'BTC',
clear: true, clear: true,
autoComplete: 'off', autoComplete: 'off',
optional: 'for receiving', placeholder: 'BTC',
serverOnly: true, serverOnly: true,
validate: string() validate: string()
.transform(value => value ? value.toUpperCase() : 'BTC') .transform(value => value ? value.toUpperCase() : 'BTC')
.oneOf(['BTC'], 'must be BTC') .oneOf(['BTC'], 'must be BTC'),
optional: 'for receiving',
requiredWithout: 'currency'
} }
] ]

View File

@ -62,6 +62,7 @@ function composeWalletSchema (walletDef, serverSide, skipGenerated) {
const { fields } = walletDef const { fields } = walletDef
const vaultEntrySchemas = { required: [], optional: [] } const vaultEntrySchemas = { required: [], optional: [] }
const cycleBreaker = []
const schemaShape = fields.reduce((acc, field) => { const schemaShape = fields.reduce((acc, field) => {
const { name, validate, optional, generated, clientOnly, requiredWithout } = field const { name, validate, optional, generated, clientOnly, requiredWithout } = field
@ -78,8 +79,14 @@ function composeWalletSchema (walletDef, serverSide, skipGenerated) {
if (!optional) { if (!optional) {
acc[name] = acc[name].required('required') acc[name] = acc[name].required('required')
} else if (requiredWithout) { } else if (requiredWithout) {
const myName = serverSide ? 'vaultEntries' : name
const partnerName = serverSide ? 'vaultEntries' : requiredWithout
// if a cycle breaker between this pair hasn't been added yet, add it
if (!cycleBreaker.some(pair => pair[1] === myName)) {
cycleBreaker.push([myName, partnerName])
}
// if we are the server, the pairSetting will be in the vaultEntries array // if we are the server, the pairSetting will be in the vaultEntries array
acc[name] = acc[name].when([serverSide ? 'vaultEntries' : requiredWithout], ([pairSetting], schema) => { acc[name] = acc[name].when([partnerName], ([pairSetting], schema) => {
if (!pairSetting || (serverSide && !pairSetting.some(v => v.key === requiredWithout))) { if (!pairSetting || (serverSide && !pairSetting.some(v => v.key === requiredWithout))) {
return schema.required(`required if ${requiredWithout} not set`) return schema.required(`required if ${requiredWithout} not set`)
} }
@ -99,9 +106,9 @@ function composeWalletSchema (walletDef, serverSide, skipGenerated) {
schemaShape.vaultEntries = Yup.array().equalto(vaultEntrySchemas) schemaShape.vaultEntries = Yup.array().equalto(vaultEntrySchemas)
} }
// we use Object.keys(schemaShape).reverse() to avoid cyclic dependencies in Yup schema // we use cycleBreaker to avoid cyclic dependencies in Yup schema
// see https://github.com/jquense/yup/issues/176#issuecomment-367352042 // see https://github.com/jquense/yup/issues/176#issuecomment-367352042
const composedSchema = Yup.object().shape(schemaShape, Object.keys(schemaShape).reverse()).concat(Yup.object({ const composedSchema = Yup.object().shape(schemaShape, cycleBreaker).concat(Yup.object({
enabled: Yup.boolean(), enabled: Yup.boolean(),
priority: Yup.number().min(0, 'must be at least 0').max(100, 'must be at most 100') priority: Yup.number().min(0, 'must be at least 0').max(100, 'must be at most 100')
})) }))