Skip to content

Latest commit

 

History

History
182 lines (161 loc) · 4.13 KB

no-invalid-state-props.md

File metadata and controls

182 lines (161 loc) · 4.13 KB

Forbid invalid properties in state node declarations

Forbid unrecognized properties in state nodes.

Rule Details

State node declarations should not contain properties which are not recognized by XState.

Furthermore, certain recognized properties are valid only within specific contexts:

  • The onDone transitions cannot be defined on the root state node. This is because onDone is a transition on a 'done.state.*' event, and when a machine reaches its final state, it can no longer accept any event.
  • The context property can be defined only on the root state node.
  • The initial property can be defined only on a compound state node. A state node is considered compound when it has type:'compound' or it has no type prop and contains a states prop.
  • The history and target properties are valid only within a history state node.

Certain properties accept specific values only:

  • type can be one of: "atomic", "compound", "parallel", "history", "final".
  • history can be one of: "shallow", "deep".

XState v5

In XState v5 some props are no longer valid on state nodes:

  • activities
  • data: removed in favor of a new prop output
  • strict
  • schema: removed in favor of a new prop types
  • tsTypes: removed in favor of a new prop types
  • preserveActionOrder
  • predictableActionArguments

If the above props are used with XState v5, this rule will report an error.

Examples of incorrect code for this rule:

// ❌
createMachine({
  states: {
    idle: {
      id: 'idle-state',
      enter: 'log', // unrecognized property name
    },
  },
})

// ❌ context can be declared only on the root state node
createMachine({
  states: {
    idle: {
      context: {}, // ???
    },
  },
})

// ❌ invalid value of the "type" prop
createMachine({
  states: {
    idle: {
      type: 'paralel', // ??? typo in the value
    },
    finished: {
      type: 'done', // ??? "done" is not a valid type
    },
  },
})

// ❌ some props are valid only in specific contexts
createMachine({
  states: {
    idle: {
      type: 'compound',
      history: 'shallow', // ??? "history" prop is valid only on history nodes
      target: 'dirty', // ??? "target" prop is valid only on history nodes
    },
    busy: {
      type: 'atomic',
      initial: 'saving', // ??? "initial" prop is valid only on compound nodes
    },
  },
})

Examples of correct code for this rule:

// ✅ all state props are valid (XState v4)
createMachine({
  id: 'root',
  strict: true,
  description: 'this is the main machine',
  predictableActionArguments: true,
  preserveActionOrder: true,
  tsTypes: {},
  schema: {},
  context: {}, // valid in the root node
  initial: 'idle',
  states: {
    idle: {
      type: 'parallel',
      entry: 'log',
      exit: 'log',
      always: [],
      after: {},
      states: {},
      onDone: {},
      on: {},
      tags: ['off'],
      activities: ['beeping'],
    },
    busy: {
      type: 'compound',
      initial: 'reading',
      states: {
        hist: {
          type: 'history',
          history: 'deep',
          target: 'writing',
        },
        reading: {
          meta: {
            value: 42,
          },
        },
        writing: {},
      },
    },
    done: {
      type: 'final',
      data: { answer: 42 },
    },
  },
})

// ✅ all state props are valid (XState v5)
createMachine({
  id: 'root',
  types: {},
  description: 'this is the main machine',
  context: {}, // valid in the root node
  initial: 'idle',
  states: {
    idle: {
      type: 'parallel',
      entry: 'log',
      exit: 'log',
      always: [],
      after: {},
      states: {},
      onDone: {},
      on: {},
      tags: ['off'],
    },
    busy: {
      type: 'compound',
      initial: 'reading',
      states: {
        hist: {
          type: 'history',
          history: 'deep',
          target: 'writing',
        },
        reading: {
          meta: {
            value: 42,
          },
        },
        writing: {},
      },
    },
    done: {
      type: 'final',
      output: { answer: 42 },
    },
  },
})

Further Reading