import { getGetServerSideProps } from '@/api/ssrApollo'
import { Form, ClientInput, ClientCheckbox, PasswordInput } from '@/components/form'
import { CenterLayout } from '@/components/layout'
import { WalletSecurityBanner } from '@/components/banners'
import { WalletLogs } from '@/components/wallet-logger'
import { useToast } from '@/components/toast'
import { useRouter } from 'next/router'
import { useWallet, Status } from 'wallets'
import Info from '@/components/info'
import Text from '@/components/text'
import { AutowithdrawSettings } from '@/components/autowithdraw-shared'
import dynamic from 'next/dynamic'
import { object, string } from 'yup'
import { autowithdrawSchemaMembers } from '@/lib/validate'
import { useMe } from '@/components/me'
import { TOR_REGEXP } from '@/lib/url'
const WalletButtonBar = dynamic(() => import('@/components/wallet-buttonbar.js'), { ssr: false })
export const getServerSideProps = getGetServerSideProps({ authRequired: true })
export default function WalletSettings () {
const toaster = useToast()
const router = useRouter()
const { wallet: name } = router.query
const wallet = useWallet(name)
const me = useMe()
const initial = wallet.fields.reduce((acc, field) => {
// We still need to run over all wallet fields via reduce
// even though we use wallet.config as the initial value
// since wallet.config is empty when wallet is not configured.
// Also, wallet.config includes general fields like
// 'enabled' and 'priority' which are not defined in wallet.fields.
return {
...acc,
[field.name]: wallet.config?.[field.name] || ''
}
}, wallet.config)
const schema = generateSchema(wallet, { me })
return (
{wallet.card.title}
{wallet.card.subtitle}
{!wallet.walletType && }
)
}
function WalletFields ({ wallet: { config, fields } }) {
return fields.map(({ name, label, type, help, optional, ...props }, i) => {
const rawProps = {
...props,
name,
initialValue: config?.[name],
label: (
{label}
{/* help can be a string or object to customize the label */}
{help && (
{help.text || help}
)}
{optional && (
{typeof optional === 'boolean' ? 'optional' : {optional}}
)}
),
required: !optional,
autoFocus: i === 0
}
if (type === 'text') {
return
}
if (type === 'password') {
return
}
return null
})
}
function generateSchema (wallet, { me }) {
if (wallet.schema) return wallet.schema
const fieldValidator = (field) => {
if (!field.validate) {
// default validation
let validator = string()
if (!field.optional) validator = validator.required('required')
return validator
}
if (field.validate.schema) {
// complex validation
return field.validate.schema
}
const { type: validationType } = field.validate
let validator
const stringTypes = ['url', 'string']
if (stringTypes.includes(validationType)) {
validator = string()
if (field.validate.length) {
validator = validator.length(field.validate.length)
}
}
if (validationType === 'url') {
validator = process.env.NODE_ENV === 'development'
? validator
.or([string().matches(/^(http:\/\/)?localhost:\d+$/), string().url()], 'invalid url')
: validator
.url()
.test(async (url, context) => {
if (field.validate.torAllowed && TOR_REGEXP.test(url)) {
// allow HTTP and HTTPS over Tor
if (!/^https?:\/\//.test(url)) {
return context.createError({ message: 'http or https required' })
}
return true
}
try {
// force HTTPS over clearnet
await string().https().validate(url)
} catch (err) {
return context.createError({ message: err.message })
}
return true
})
}
if (!field.optional) validator = validator.required('required')
return validator
}
return object(
wallet.fields.reduce((acc, field) => {
return {
...acc,
[field.name]: fieldValidator(field)
}
}, wallet.walletType ? autowithdrawSchemaMembers({ me }) : {})
)
}