|
|
|
#!/usr/bin/env node
|
|
|
|
|
|
|
|
const ansiEscapes = require('ansi-escapes')
|
|
|
|
const chalk = require('chalk')
|
|
|
|
const ccValidator = require('credit-card')
|
|
|
|
|
|
|
|
const textInput = require('../lib/utils/input/text')
|
|
|
|
const countries = require('../lib/utils/billing/country-list')
|
|
|
|
const cardBrands = require('../lib/utils/billing/card-brands')
|
|
|
|
const geocode = require('../lib/utils/billing/geocode')
|
|
|
|
const success = require('../lib/utils/output/success')
|
|
|
|
const wait = require('../lib/utils/output/wait')
|
|
|
|
|
|
|
|
function rightPad(string, n = 12) {
|
|
|
|
n -= string.length
|
|
|
|
return string + ' '.repeat(n > -1 ? n : 0)
|
|
|
|
}
|
|
|
|
|
|
|
|
function expDateMiddleware(data) {
|
|
|
|
return data
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = function (creditCards) {
|
|
|
|
const state = {
|
|
|
|
error: undefined,
|
|
|
|
cardGroupLabel: `> ${chalk.bold('Enter your card details')}`,
|
|
|
|
|
|
|
|
name: {
|
|
|
|
label: rightPad('Name'),
|
|
|
|
placeholder: 'John Appleseed',
|
|
|
|
validateValue: data => data.trim().length > 0
|
|
|
|
},
|
|
|
|
|
|
|
|
cardNumber: {
|
|
|
|
label: rightPad('Number'),
|
|
|
|
mask: 'cc',
|
|
|
|
placeholder: '#### #### #### ####',
|
|
|
|
validateKeypress: (data, value) => (
|
|
|
|
/\d/.test(data) && value.length < 19
|
|
|
|
),
|
|
|
|
validateValue: data => {
|
|
|
|
data = data.replace(/ /g, '')
|
|
|
|
const type = ccValidator.determineCardType(data)
|
|
|
|
if (!type) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return ccValidator.isValidCardNumber(data, type)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
ccv: {
|
|
|
|
label: rightPad('CCV'),
|
|
|
|
mask: 'ccv',
|
|
|
|
placeholder: '###',
|
|
|
|
validateValue: data => {
|
|
|
|
const brand = state.cardNumber.brand.toLowerCase()
|
|
|
|
return ccValidator.doesCvvMatchType(data, brand)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
expDate: {
|
|
|
|
label: rightPad('Exp. Date'),
|
|
|
|
mask: 'expDate',
|
|
|
|
placeholder: 'mm / yyyy',
|
|
|
|
middleware: expDateMiddleware,
|
|
|
|
validateValue: data => !ccValidator.isExpired(...data.split(' / '))
|
|
|
|
},
|
|
|
|
|
|
|
|
addressGroupLabel: `\n> ${chalk.bold('Enter your billing address')}`,
|
|
|
|
|
|
|
|
country: {
|
|
|
|
label: rightPad('Country'),
|
|
|
|
async autoComplete(value) {
|
|
|
|
for (const country in countries) {
|
|
|
|
if (!Object.hasOwnProperty.call(countries, country)) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if (country.startsWith(value)) {
|
|
|
|
return country.substr(value.length)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
},
|
|
|
|
validateValue: value => countries[value] !== undefined
|
|
|
|
},
|
|
|
|
|
|
|
|
zipCode: {
|
|
|
|
label: rightPad('ZIP'),
|
|
|
|
validadeKeypress: data => data.trim().length > 0,
|
|
|
|
validateValue: data => data.trim().length > 0
|
|
|
|
},
|
|
|
|
|
|
|
|
state: {
|
|
|
|
label: rightPad('State'),
|
|
|
|
validateValue: data => data.trim().length > 0
|
|
|
|
},
|
|
|
|
|
|
|
|
city: {
|
|
|
|
label: rightPad('City'),
|
|
|
|
validateValue: data => data.trim().length > 0
|
|
|
|
},
|
|
|
|
|
|
|
|
address1: {
|
|
|
|
label: rightPad('Address'),
|
|
|
|
validateValue: data => data.trim().length > 0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async function render() {
|
|
|
|
for (const key in state) {
|
|
|
|
if (!Object.hasOwnProperty.call(state, key)) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
const piece = state[key]
|
|
|
|
if (typeof piece === 'string') {
|
|
|
|
console.log(piece)
|
|
|
|
} else if (typeof piece === 'object') {
|
|
|
|
let result
|
|
|
|
try {
|
|
|
|
result = await textInput({
|
|
|
|
label: '- ' + piece.label,
|
|
|
|
initialValue: piece.initialValue || piece.value,
|
|
|
|
placeholder: piece.placeholder,
|
|
|
|
mask: piece.mask,
|
|
|
|
validateKeypress: piece.validateKeypress,
|
|
|
|
validateValue: piece.validateValue,
|
|
|
|
autoComplete: piece.autoComplete
|
|
|
|
})
|
|
|
|
piece.value = result
|
|
|
|
if (key === 'cardNumber') {
|
|
|
|
let brand = cardBrands[ccValidator.determineCardType(result)]
|
|
|
|
piece.brand = brand
|
|
|
|
if (brand === 'American Express') {
|
|
|
|
state.ccv.placeholder = '#'.repeat(4)
|
|
|
|
} else {
|
|
|
|
state.ccv.placeholder = '#'.repeat(3)
|
|
|
|
}
|
|
|
|
brand = chalk.cyan(`[${brand}]`)
|
|
|
|
const masked = chalk.gray('#### '.repeat(3)) + result.split(' ')[3]
|
|
|
|
process.stdout.write(
|
|
|
|
`${chalk.cyan('✓')} ${piece.label}${masked} ${brand}\n`
|
|
|
|
)
|
|
|
|
} else if (key === 'ccv') {
|
|
|
|
process.stdout.write(
|
|
|
|
`${chalk.cyan('✓')} ${piece.label}${'*'.repeat(result.length)}\n`
|
|
|
|
)
|
|
|
|
} else if (key === 'expDate') {
|
|
|
|
let text = result.split(' / ')
|
|
|
|
text = text[0] + chalk.gray(' / ') + text[1]
|
|
|
|
process.stdout.write(`${chalk.cyan('✓')} ${piece.label}${text}\n`)
|
|
|
|
} else if (key === 'zipCode') {
|
|
|
|
const stopSpinner = wait(piece.label + result)
|
|
|
|
const addressInfo = await geocode({
|
|
|
|
country: state.country.value,
|
|
|
|
zipCode: result
|
|
|
|
})
|
|
|
|
if (addressInfo.state) {
|
|
|
|
state.state.initialValue = addressInfo.state
|
|
|
|
}
|
|
|
|
if (addressInfo.city) {
|
|
|
|
state.city.initialValue = addressInfo.city
|
|
|
|
}
|
|
|
|
stopSpinner()
|
|
|
|
process.stdout.write(`${chalk.cyan('✓')} ${piece.label}${result}\n`)
|
|
|
|
} else {
|
|
|
|
process.stdout.write(`${chalk.cyan('✓')} ${piece.label}${result}\n`)
|
|
|
|
}
|
|
|
|
} catch (err) {
|
|
|
|
if (err.message === 'USER_ABORT') {
|
|
|
|
process.exit(1)
|
|
|
|
} else {
|
|
|
|
console.error(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
console.log('') // new line
|
|
|
|
const stopSpinner = wait('Saving card')
|
|
|
|
|
|
|
|
try {
|
|
|
|
const res = await creditCards.add({
|
|
|
|
name: state.name.value,
|
|
|
|
cardNumber: state.cardNumber.value,
|
|
|
|
ccv: state.ccv.value,
|
|
|
|
expDate: state.expDate.value,
|
|
|
|
country: state.country.value,
|
|
|
|
zipCode: state.zipCode.value,
|
|
|
|
state: state.state.value,
|
|
|
|
city: state.city.value,
|
|
|
|
address1: state.address1.value
|
|
|
|
})
|
|
|
|
stopSpinner()
|
|
|
|
success(`${state.cardNumber.brand} ending in ${res.last4} was added to your account`)
|
|
|
|
} catch (err) {
|
|
|
|
stopSpinner()
|
|
|
|
const linesToClear = state.error ? 13 : 12
|
|
|
|
process.stdout.write(ansiEscapes.eraseLines(linesToClear))
|
|
|
|
state.error = `${chalk.red('> Error!')} ${err.message} Please make sure the info is correct`
|
|
|
|
await render()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
render().catch(console.error)
|
|
|
|
}
|