ember-cli-typescript icon indicating copy to clipboard operation
ember-cli-typescript copied to clipboard

`AsyncHasMany` resolves to `ArrayProxy`, can't access `reload`/`createRecord` methods on resolved promise

Open charlesfries opened this issue 3 years ago • 2 comments

Which package(s) does this problem pertain to?

  • [X] @types/ember-data
"ember-cli-typescript": "^5.1.1",
"ember-source": "~4.7.0",
"typescript": "^4.8.3",

Using latest types

What are instructions we can follow to reproduce the issue?

With the recent changes to store.query and ArrayProxy the following snippet seems to be completely type safe and valid:

const posts = await this.store.query('post', {}); // resolves to AdapterPopulatedRecordArray
                                                  // in my case this promise resolution is happening in model hook and update is in controller action
await posts.update(); // OK, no any leakage

My issue arises when I try to do a similar operation with an AsyncHasMany relationship using these updated types:

const post = await this.store.findRecord('post', 1);

await post.comments.reload(); // OK, post.comments is AsyncHasMany
post.comments.createRecord(); // OK, post.comments is AsyncHasMany

const comments = await post.comments; // resolves to ArrayProxy
                                      // this is all that I return from model hook, so I don't have access to unresolved relationship

await comments.reload(); // DNE on ArrayProxy
comments.createRecord(); // DNE on ArrayProxy

Expectation: reload/createRecord access on resolved relationship Reality: reload/createRecord do not exist on resolved AsyncHasMany relationship type ArrayProxy

My question is: am I errantly accessing reload/createRecord on this resolved relationship, or is this a typing issue?

If the latter, what does PromiseManyArray need to resolve to to retain these methods? This is the current type:

type AsyncHasMany<T extends Model> = PromiseManyArray<T>;

class PromiseManyArray<T extends Model> extends PromiseArray<T, Ember.ArrayProxy<T>> {
  reload(): PromiseManyArray<T>;
  createRecord(inputProperties?: {}): T;
}

charlesfries avatar Sep 21 '22 18:09 charlesfries

It’s likely a types issue. The types for these haven’t meaningfully changed in a very long time. It may be best to tackle as part of the work we’re about to spin up for publishing preview types directly from the Ember Data repo (analogous to what we’re doing in Ember).

chriskrycho avatar Sep 22 '22 13:09 chriskrycho

We are facing a similar issue. What is the correct way to work around this problem?

What I tried is the following. (I try to illustrate it with pseudo-code and the previous example):

export const fixMeEmberDataHasManyToArray = <T>(input: unknown) => input as T[];
export const fixMeEmberDataHasManyToHasMany =  = <T extends Model>(input: unknown) => input as SyncHasMany<T>;
const comments = fixMeEmberDataHasManyToHasMany<CommentModel>(await post.comments);

await comments.reload(); // DNE on ArrayProxy
comments.createRecord(); // DNE on ArrayProxy

But that's a little bit tedious 😬 on the other hand, we can "easily" identify the problematic code sections later if the problems are fixed. But of course, it's just a hack. Therefore it would be interesting to get the opinions of the pros on what to do

tschoartschi avatar Jan 12 '24 08:01 tschoartschi