Skip to content

Instantly share code, notes, and snippets.

@TheDutchCoder
Last active September 24, 2020 13:38
Show Gist options
  • Save TheDutchCoder/345c4d4280c5e1d9b736bfc6afee0665 to your computer and use it in GitHub Desktop.
Save TheDutchCoder/345c4d4280c5e1d9b736bfc6afee0665 to your computer and use it in GitHub Desktop.
Generated by XState Viz: https://xstate.js.org/viz
const sendCode = (email) => {
let error = Math.random() < 0.25
return new Promise((resolve, reject) => {
setTimeout(() => {
if (error) {
reject({ code: 3 })
}
if (email === '[email protected]') {
resolve()
}
reject({ code: 1 })
}, 1500)
})
}
const verifyCode = (code) => {
let error = Math.random() < 0.25
let expired = Math.random() < 0.25
return new Promise((resolve, reject) => {
setTimeout(() => {
if (error) {
reject({ code: 3 })
}
if (expired) {
reject({ code: 2 })
}
if (code === '12345') {
resolve()
}
reject({ code: 1 })
}, 1500)
})
}
const emailStates = {
initial: 'noError',
states: {
noError: {},
error: {
initial: 'empty',
states: {
empty: {},
badFormat: {},
},
},
},
}
const authStates = {
initial: 'noError',
states: {
noError: {},
error: {
initial: 'internal',
states: {
internal: {},
},
},
},
}
const codeStates = {
initial: 'noError',
states: {
noError: {},
error: {
initial: 'empty',
states: {
empty: {},
badFormat: {},
expired: {},
},
},
},
}
const signupMachine = Machine({
id: 'signup',
initial: 'details',
context: {
email: '',
code: '',
},
states: {
details: {
initial: 'ready',
states: {
ready: {
type: 'parallel',
on: {
SUBMIT: [
{
cond: 'isEmailEmpty',
target: 'ready.email.error.empty',
},
{
cond: 'isEmailBadFormat',
target: 'ready.email.error.badFormat',
},
{ target: 'waiting' },
],
INPUT_EMAIL: {
actions: 'cacheEmail',
target: 'ready.email',
},
TOGGLE: '#signup.verify',
},
initial: 'noError',
states: {
email: { ...emailStates },
auth: { ...authStates },
},
},
waiting: {
invoke: {
src: 'handleEmail',
onDone: '#signup.verify.ready',
onError: [
{
cond: 'isEmailEmpty',
target: 'ready.email.error.empty',
},
{
cond: 'isEmailBadFormat',
target: 'ready.email.error.badFormat',
},
{
cond: 'isServerError',
target: 'ready.auth.error.internal',
},
],
},
},
},
},
verify: {
initial: 'ready',
states: {
ready: {
type: 'parallel',
on: {
SUBMIT: [
{
cond: 'isCodeEmpty',
target: 'ready.code.error.empty',
},
{
cond: 'isCodeBadFormat',
target: 'ready.code.error.badFormat',
},
{ target: 'waiting' },
],
INPUT_CODE: {
actions: 'cacheCode',
target: 'ready',
},
TOGGLE: '#signup.details',
},
initial: 'noError',
states: {
code: { ...codeStates },
auth: { ...authStates },
},
},
waiting: {
invoke: {
src: 'handleCode',
onDone: '#signup.dashboard',
onError: [
{
cond: 'isCodeEmpty',
target: 'ready.code.error.empty',
},
{
cond: 'isCodeBadFormat',
target: 'ready.code.error.badFormat',
},
{
cond: 'isCodeExpired',
target: 'ready.code.error.expired',
},
{
cond: 'isServerError',
target: 'ready.auth.error.internal',
},
],
},
},
},
},
dashboard: { type: 'final' }
},
}, {
guards: {
isEmailEmpty: (ctx, evt) => ctx.email === '',
isEmailBadFormat: (ctx, evt) => evt.data && evt.data.code === 1,
isCodeEmpty: (ctx, evt) => ctx.code === '',
isCodeBadFormat: (ctx, evt) => evt.data && evt.data.code === 1,
isCodeExpired: (ctx, evt) => evt.data && evt.data.code === 2,
isServerError: (ctx, evt) => evt.data && evt.data.code === 3,
},
services: {
handleEmail: (ctx, evt) => sendCode(ctx.email),
handleCode: (ctx, evt) => verifyCode(ctx.code),
},
actions: {
cacheEmail: assign((ctx, evt) => ({
email: evt.email
})),
cacheCode: assign((ctx, evt) => ({
code: evt.code
})),
},
})
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment