state.go 1.5 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970
  1. package lz4
  2. import (
  3. "errors"
  4. "fmt"
  5. "io"
  6. "github.com/pierrec/lz4/internal/lz4errors"
  7. )
  8. //go:generate go run golang.org/x/tools/cmd/stringer -type=aState -output state_gen.go
  9. const (
  10. noState aState = iota // uninitialized reader
  11. errorState // unrecoverable error encountered
  12. newState // instantiated object
  13. readState // reading data
  14. writeState // writing data
  15. closedState // all done
  16. )
  17. type (
  18. aState uint8
  19. _State struct {
  20. states []aState
  21. state aState
  22. err error
  23. }
  24. )
  25. func (s *_State) init(states []aState) {
  26. s.states = states
  27. s.state = states[0]
  28. }
  29. // next sets the state to the next one unless it is passed a non nil error.
  30. // It returns whether or not it is in error.
  31. func (s *_State) next(err error) bool {
  32. if err != nil {
  33. s.err = fmt.Errorf("%s: %w", s.state, err)
  34. s.state = errorState
  35. return true
  36. }
  37. s.state = s.states[s.state]
  38. return false
  39. }
  40. // nextd is like next but for defers.
  41. func (s *_State) nextd(errp *error) bool {
  42. return errp != nil && s.next(*errp)
  43. }
  44. // check sets s in error if not already in error and if the error is not nil or io.EOF,
  45. func (s *_State) check(errp *error) {
  46. if s.state == errorState || errp == nil {
  47. return
  48. }
  49. if err := *errp; err != nil {
  50. s.err = fmt.Errorf("%w[%s]", err, s.state)
  51. if !errors.Is(err, io.EOF) {
  52. s.state = errorState
  53. }
  54. }
  55. }
  56. func (s *_State) fail() error {
  57. s.state = errorState
  58. s.err = fmt.Errorf("%w[%s]", lz4errors.ErrInternalUnhandledState, s.state)
  59. return s.err
  60. }