Decorator field/accessor initializer order

Chris Hewell Garrett

2023-05

  • Field and accessor initializers run from innermost decorator to outermost decorator
  • Getters/setters/methods are replaced from innermost decorator to outermost decorator
  • This means that getters/setters/methods execute from outermost to innermost
  • This makes it impossible currently to have initial values match assigned value for certain accessors
function minusTwo({ set }) {
  return {
    set(v) {
      set.call(this, v - 2)
    },
    init(v) {
      return v - 2;
    }
  }
}

function timesFour({ set }) {
  return {
    set(v) {
      set.call(this, v * 4)
    },
    init(v) {
      return v * 4;
    }
  }
}

class Foo {
  @minusTwo @timesFour accessor bar = 5;
}

const foo = new Foo();
console.log(foo.bar); // 18

foo.bar = 5;
console.log(foo.bar); // 12 

Why is this an issue now?

  • Previously, TypeScript and Babel legacy did run initializers in the current order, innermost -> outermost
  • However, they also assigned the value with [[Set]] semantics to the field, thus running it through all of the setters for the field 
  • The switch to [[Define]] semantics for fields + the redesign of accessors led to this being an issue that wasn't realized until it had been used in the ecosystem for some time

Proposed Solution 1

Reverse the order of initializers, run them from outermost -> innermost

  • Conceptually, we are running the initializers as if the value is being [[Set]] on initialization, so it runs through them in the same order as method calls.
  • This means initialization and method calls always execute from outermost to innermost
  • Decorators still EVALUATE from innermost -> outermost. That does not change.
  • Allows users to distinguish between initial value and updated value, for auto-accessors
class Foo {
  @minusTwo @timesFour accessor bar = 5;
}

// 5 - 2 * 4 = 12

const foo = new Foo();
console.log(foo.bar); // 12

foo.bar = 5;
console.log(foo.bar); // 12 

Proposed Solution 2

Restore the previous behavior, have setters called with initial value for auto-accessors only

  • Most similar to existing behavior in the ecosystem, so less churn overall
  • Can still distinguish between initial and updated value, but requires a flag to do so (e.g. via WeakSet)
  • Only affects auto-accessors, field/method behavior remains the same
class Foo {
  @minusTwo @timesFour accessor bar = 5;
}

// Initial value = 4 * 5 - 2 = 18
// Set value = Initial value - 2 * 4 = 64

const foo = new Foo();
console.log(foo.bar); // 64

foo.bar = 5;
console.log(foo.bar); // 12 

Decorator field/accessor initializer order

By pzuraq

Decorator field/accessor initializer order

  • 583