From cdd04dab708955e862166ce1a00bb2a74b06f26b Mon Sep 17 00:00:00 2001 From: Chris Krycho Date: Mon, 7 Nov 2022 20:42:04 -0700 Subject: [PATCH 1/3] Add failing type test for `getOwner(Component)` This catches the scenario where we have a component which actually has a signature defined: class Example extends Component<{ Args: { anyWillDo: true } }> { method() { getOwner(this); // should be `Owner`, not `| undefined` } } Presently, we return `Owner | undefined` instead. --- type-tests/preview/@ember/owner-tests.ts | 28 ++++++++++++++++++++++++ 1 file changed, 28 insertions(+) diff --git a/type-tests/preview/@ember/owner-tests.ts b/type-tests/preview/@ember/owner-tests.ts index 15ad10010d0..83be87e1161 100644 --- a/type-tests/preview/@ember/owner-tests.ts +++ b/type-tests/preview/@ember/owner-tests.ts @@ -6,7 +6,11 @@ import Owner, { Resolver, KnownForTypeResult, } from '@ember/owner'; +import Component from '@glimmer/component'; import { expectTypeOf } from 'expect-type'; +// TODO: once we move the runtime export to `@ember/owner`, update this import +// as well. (That's why it's tested in this module!) +import { getOwner } from '@ember/application'; // Just a class we can construct in the Factory and FactoryManager tests declare class ConstructThis { @@ -143,6 +147,30 @@ typedStrings.map((aString) => owner.lookup(aString)); const aConstName = 'type:name'; expectTypeOf(owner.lookup(aConstName)).toBeUnknown(); +// Check handling with Glimmer components carrying a Signature: they should +// properly resolve to `Owner`, *not* `Owner | undefined`. +interface Sig { + Args: { + name: string; + age: number; + extra: T; + }; + Element: HTMLParagraphElement; + Blocks: { + default: [greeting: string]; + extra: [T]; + }; +} + +class ExampleComponent extends Component> { + checkThis() { + expectTypeOf(getOwner(this)).toEqualTypeOf(); + } +} + +declare let example: ExampleComponent; +expectTypeOf(getOwner(example)).toEqualTypeOf(); + // ----- Minimal further coverage for POJOs ----- // // `Factory` and `FactoryManager` don't have to deal in actual classes. :sigh: const Creatable = { From fc09d4944e031ba0c7350b2249539daf365ef1f9 Mon Sep 17 00:00:00 2001 From: Chris Krycho Date: Mon, 7 Nov 2022 20:43:44 -0700 Subject: [PATCH 2/3] Fix type definition of `getOwner`, and explain it Use `GlimmerComponent` instead of plain `GlimmerComponent`, since the latter resolves to the default `GlimmerComponent`, which ends up being too narrow to resolve correctly with `getOwner` when any actual signature is applied. As a bonus, rename `FrameworkObject` in this context to the somewhat more verbose `KnownFrameworkObject`, which avoids a conflict with the actual `FrameworkObject` type used internally. --- types/preview/@ember/application/index.d.ts | 9 ++++++++- 1 file changed, 8 insertions(+), 1 deletion(-) diff --git a/types/preview/@ember/application/index.d.ts b/types/preview/@ember/application/index.d.ts index 9fe576425c9..6637a61ecd3 100644 --- a/types/preview/@ember/application/index.d.ts +++ b/types/preview/@ember/application/index.d.ts @@ -122,7 +122,13 @@ declare module '@ember/application' { // whenever we add new base classes to the framework. For example, if we // introduce a standalone `Service` or `Route` base class which *does not* // extend from `EmberObject`, it will need to be added here. - type FrameworkObject = EmberObject | GlimmerComponent; + // + // NOTE: we use `any` here because we need to make sure *not* to fix the + // actual GlimmerComponent type; using `unknown` or `{}` or `never` (the + // obvious alternatives here) results in a version which is too narrow, such + // that any subclass which applies a signature does not get resolved by the + // definition of `getOwner()` below. + type KnownFrameworkObject = EmberObject | GlimmerComponent; /** * Framework objects in an Ember application (components, services, routes, etc.) @@ -131,6 +137,7 @@ declare module '@ember/application' { * instantiation and manages its lifetime. */ export function getOwner(object: FrameworkObject): Owner; + export function getOwner(object: KnownFrameworkObject): Owner; export function getOwner(object: unknown): Owner | undefined; /** * `setOwner` forces a new owner on a given object instance. This is primarily From dff8831cdbb5583de443a1859e4f09054687119f Mon Sep 17 00:00:00 2001 From: Chris Krycho Date: Mon, 7 Nov 2022 20:45:44 -0700 Subject: [PATCH 3/3] Explain the (un)safety of the type of `getOwner()` The type for `getOwner()` is *technically* unsafe, but this unsafety is one which end users will rarely hit in practice and, courtesy of Ember's current types, is unavoidable if `getOwner(someService)` is to correctly return `Owner` instead of `Owner | undefined`. The Typed Ember folks have discussed this point at some length, but I realized on looking at this to make the changes in the preceding two commits that we didn't have it committed to the code base anywhere, so here I am just introducing a long comment explaining it. When we switch from preview to stable types, this comment should come along! --- types/preview/@ember/application/index.d.ts | 9 ++++++++- 1 file changed, 8 insertions(+), 1 deletion(-) diff --git a/types/preview/@ember/application/index.d.ts b/types/preview/@ember/application/index.d.ts index 6637a61ecd3..0543ff53508 100644 --- a/types/preview/@ember/application/index.d.ts +++ b/types/preview/@ember/application/index.d.ts @@ -136,7 +136,14 @@ declare module '@ember/application' { * objects is the responsibility of an "owner", which handled its * instantiation and manages its lifetime. */ - export function getOwner(object: FrameworkObject): Owner; + // SAFETY: this first overload is, strictly speaking, *unsafe*. It is possible + // to do `let x = EmberObject.create(); getOwner(x);` and the result will *not* + // be `Owner` but instead `undefined`. However, that's quite unusual at this + // point, and more to the point we cannot actually distinguish a `Service` + // subclass from `EmberObject` at this point: `Service` subclasses `EmberObject` + // and adds nothing to it. Accordingly, if we want to catch `Service`s with this + // (and `getOwner(this)` for some service will definitely be defined!), it has + // to be this way. :sigh: export function getOwner(object: KnownFrameworkObject): Owner; export function getOwner(object: unknown): Owner | undefined; /**