Documentation / @ember-data/model / index / hasMany
Call Signature
function hasMany(): never;
Defined in: packages/model/src/-private/has-many.ts:248
hasMany
is used to define Many-To-One and Many-To-Many, and Many-To-None relationships on a Model.
hasMany
takes a configuration hash as a second parameter, currently supported options are:
async
: (required) A boolean value used to declare whether this is a sync (false) or async (true) relationship.inverse
: (required) A string used to identify the inverse property on a related model, ornull
.polymorphic
: (optional) A boolean value to mark the relationship as polymorphicas
: (optional) A string used to declare the abstract type "this" record satisfies for polymorphism.
Examples
To declare a many-to-one (or one-to-many) relationship, use belongsTo
in combination with hasMany
:
// app/models/post.js
import Model, { hasMany } from '@ember-data/model';
export default class Post extends Model {
@hasMany('comment', { async: false, inverse: 'post' }) comments;
}
// app/models/comment.js
import Model, { belongsTo } from '@ember-data/model';
export default class Comment extends Model {
@belongsTo('post', { async: false, inverse: 'comments' }) post;
}
To declare a many-to-many relationship with managed inverses, use hasMany
for both sides:
// app/models/post.js
import Model, { hasMany } from '@ember-data/model';
export default class Post extends Model {
@hasMany('tag', { async: true, inverse: 'posts' }) tags;
}
// app/models/tag.js
import Model, { hasMany } from '@ember-data/model';
export default class Tag extends Model {
@hasMany('post', { async: true, inverse: 'tags' }) posts;
}
To declare a many-to-many relationship without managed inverses, use hasMany
for both sides with null
as the inverse:
// app/models/post.js
import Model, { hasMany } from '@ember-data/model';
export default class Post extends Model {
@hasMany('tag', { async: true, inverse: null }) tags;
}
// app/models/tag.js
import Model, { hasMany } from '@ember-data/model';
export default class Tag extends Model {
@hasMany('post', { async: true, inverse: null }) posts;
}
To declare a many-to-none relationship between two models, use hasMany
with inverse set to null
on just one side::
// app/models/post.js
import Model, { hasMany } from '@ember-data/model';
export default class Post extends Model {
@hasMany('category', { async: true, inverse: null }) categories;
}
Sync vs Async Relationships
EmberData fulfills relationships using resource data available in the cache.
Sync relationships point directly to the known related resources.
When a relationship is declared as async, if any of the known related resources have not been loaded, they will be fetched. The property on the record when accessed provides a promise that resolves once all resources are loaded.
Async relationships may take advantage of links. On access, if the related link has not been loaded, or if any known resources are not available in the cache, the fresh state will be fetched using the link.
In contrast to async relationship, accessing a sync relationship will error on access when any of the known related resources have not been loaded.
If you are using links
with sync relationships, you have to use the HasMany reference API to fetch or refresh related resources that aren't loaded. For instance, for a comments
relationship:
post.hasMany('comments').reload();
Polymorphic Relationships
To declare a polymorphic relationship, use hasMany
with the polymorphic
option set to true
:
// app/models/comment.js
import Model, { belongsTo } from '@ember-data/model';
export default class Comment extends Model {
@belongsTo('commentable', { async: false, inverse: 'comments', polymorphic: true }) parent;
}
'commentable'
here is referred to as the "abstract type" for the polymorphic relationship.
Polymorphic relationships with inverse: null
will accept any type of record as their content. Polymorphic relationships with inverse
set to a string will only accept records with a matching inverse relationships declaring itself as satisfying the abstract type.
Below, 'as' is used to declare the that 'post' record satisfies the abstract type 'commentable' for this relationship.
// app/models/post.js
import Model, { hasMany } from '@ember-data/model';
export default class Post extends Model {
@hasMany('comment', { async: false, inverse: 'parent', as: 'commentable' }) comments;
}
Note: every Model that declares an inverse to a polymorphic relationship must declare itself exactly the same. This is because polymorphism is based on structural traits.
Polymorphic to polymorphic relationships are supported. Both sides of the relationship must be declared as polymorphic, and the as
option must be used to declare the abstract type each record satisfies on both sides.
Returns
never
relationship
Call Signature
function hasMany(type): never;
Defined in: packages/model/src/-private/has-many.ts:249
hasMany
is used to define Many-To-One and Many-To-Many, and Many-To-None relationships on a Model.
hasMany
takes a configuration hash as a second parameter, currently supported options are:
async
: (required) A boolean value used to declare whether this is a sync (false) or async (true) relationship.inverse
: (required) A string used to identify the inverse property on a related model, ornull
.polymorphic
: (optional) A boolean value to mark the relationship as polymorphicas
: (optional) A string used to declare the abstract type "this" record satisfies for polymorphism.
Examples
To declare a many-to-one (or one-to-many) relationship, use belongsTo
in combination with hasMany
:
// app/models/post.js
import Model, { hasMany } from '@ember-data/model';
export default class Post extends Model {
@hasMany('comment', { async: false, inverse: 'post' }) comments;
}
// app/models/comment.js
import Model, { belongsTo } from '@ember-data/model';
export default class Comment extends Model {
@belongsTo('post', { async: false, inverse: 'comments' }) post;
}
To declare a many-to-many relationship with managed inverses, use hasMany
for both sides:
// app/models/post.js
import Model, { hasMany } from '@ember-data/model';
export default class Post extends Model {
@hasMany('tag', { async: true, inverse: 'posts' }) tags;
}
// app/models/tag.js
import Model, { hasMany } from '@ember-data/model';
export default class Tag extends Model {
@hasMany('post', { async: true, inverse: 'tags' }) posts;
}
To declare a many-to-many relationship without managed inverses, use hasMany
for both sides with null
as the inverse:
// app/models/post.js
import Model, { hasMany } from '@ember-data/model';
export default class Post extends Model {
@hasMany('tag', { async: true, inverse: null }) tags;
}
// app/models/tag.js
import Model, { hasMany } from '@ember-data/model';
export default class Tag extends Model {
@hasMany('post', { async: true, inverse: null }) posts;
}
To declare a many-to-none relationship between two models, use hasMany
with inverse set to null
on just one side::
// app/models/post.js
import Model, { hasMany } from '@ember-data/model';
export default class Post extends Model {
@hasMany('category', { async: true, inverse: null }) categories;
}
Sync vs Async Relationships
EmberData fulfills relationships using resource data available in the cache.
Sync relationships point directly to the known related resources.
When a relationship is declared as async, if any of the known related resources have not been loaded, they will be fetched. The property on the record when accessed provides a promise that resolves once all resources are loaded.
Async relationships may take advantage of links. On access, if the related link has not been loaded, or if any known resources are not available in the cache, the fresh state will be fetched using the link.
In contrast to async relationship, accessing a sync relationship will error on access when any of the known related resources have not been loaded.
If you are using links
with sync relationships, you have to use the HasMany reference API to fetch or refresh related resources that aren't loaded. For instance, for a comments
relationship:
post.hasMany('comments').reload();
Polymorphic Relationships
To declare a polymorphic relationship, use hasMany
with the polymorphic
option set to true
:
// app/models/comment.js
import Model, { belongsTo } from '@ember-data/model';
export default class Comment extends Model {
@belongsTo('commentable', { async: false, inverse: 'comments', polymorphic: true }) parent;
}
'commentable'
here is referred to as the "abstract type" for the polymorphic relationship.
Polymorphic relationships with inverse: null
will accept any type of record as their content. Polymorphic relationships with inverse
set to a string will only accept records with a matching inverse relationships declaring itself as satisfying the abstract type.
Below, 'as' is used to declare the that 'post' record satisfies the abstract type 'commentable' for this relationship.
// app/models/post.js
import Model, { hasMany } from '@ember-data/model';
export default class Post extends Model {
@hasMany('comment', { async: false, inverse: 'parent', as: 'commentable' }) comments;
}
Note: every Model that declares an inverse to a polymorphic relationship must declare itself exactly the same. This is because polymorphism is based on structural traits.
Polymorphic to polymorphic relationships are supported. Both sides of the relationship must be declared as polymorphic, and the as
option must be used to declare the abstract type each record satisfies on both sides.
Parameters
type
string
(optional) the name of the related resource
Returns
never
relationship
Call Signature
function hasMany<T>(type, options): RelationshipDecorator<T>;
Defined in: packages/model/src/-private/has-many.ts:250
hasMany
is used to define Many-To-One and Many-To-Many, and Many-To-None relationships on a Model.
hasMany
takes a configuration hash as a second parameter, currently supported options are:
async
: (required) A boolean value used to declare whether this is a sync (false) or async (true) relationship.inverse
: (required) A string used to identify the inverse property on a related model, ornull
.polymorphic
: (optional) A boolean value to mark the relationship as polymorphicas
: (optional) A string used to declare the abstract type "this" record satisfies for polymorphism.
Examples
To declare a many-to-one (or one-to-many) relationship, use belongsTo
in combination with hasMany
:
// app/models/post.js
import Model, { hasMany } from '@ember-data/model';
export default class Post extends Model {
@hasMany('comment', { async: false, inverse: 'post' }) comments;
}
// app/models/comment.js
import Model, { belongsTo } from '@ember-data/model';
export default class Comment extends Model {
@belongsTo('post', { async: false, inverse: 'comments' }) post;
}
To declare a many-to-many relationship with managed inverses, use hasMany
for both sides:
// app/models/post.js
import Model, { hasMany } from '@ember-data/model';
export default class Post extends Model {
@hasMany('tag', { async: true, inverse: 'posts' }) tags;
}
// app/models/tag.js
import Model, { hasMany } from '@ember-data/model';
export default class Tag extends Model {
@hasMany('post', { async: true, inverse: 'tags' }) posts;
}
To declare a many-to-many relationship without managed inverses, use hasMany
for both sides with null
as the inverse:
// app/models/post.js
import Model, { hasMany } from '@ember-data/model';
export default class Post extends Model {
@hasMany('tag', { async: true, inverse: null }) tags;
}
// app/models/tag.js
import Model, { hasMany } from '@ember-data/model';
export default class Tag extends Model {
@hasMany('post', { async: true, inverse: null }) posts;
}
To declare a many-to-none relationship between two models, use hasMany
with inverse set to null
on just one side::
// app/models/post.js
import Model, { hasMany } from '@ember-data/model';
export default class Post extends Model {
@hasMany('category', { async: true, inverse: null }) categories;
}
Sync vs Async Relationships
EmberData fulfills relationships using resource data available in the cache.
Sync relationships point directly to the known related resources.
When a relationship is declared as async, if any of the known related resources have not been loaded, they will be fetched. The property on the record when accessed provides a promise that resolves once all resources are loaded.
Async relationships may take advantage of links. On access, if the related link has not been loaded, or if any known resources are not available in the cache, the fresh state will be fetched using the link.
In contrast to async relationship, accessing a sync relationship will error on access when any of the known related resources have not been loaded.
If you are using links
with sync relationships, you have to use the HasMany reference API to fetch or refresh related resources that aren't loaded. For instance, for a comments
relationship:
post.hasMany('comments').reload();
Polymorphic Relationships
To declare a polymorphic relationship, use hasMany
with the polymorphic
option set to true
:
// app/models/comment.js
import Model, { belongsTo } from '@ember-data/model';
export default class Comment extends Model {
@belongsTo('commentable', { async: false, inverse: 'comments', polymorphic: true }) parent;
}
'commentable'
here is referred to as the "abstract type" for the polymorphic relationship.
Polymorphic relationships with inverse: null
will accept any type of record as their content. Polymorphic relationships with inverse
set to a string will only accept records with a matching inverse relationships declaring itself as satisfying the abstract type.
Below, 'as' is used to declare the that 'post' record satisfies the abstract type 'commentable' for this relationship.
// app/models/post.js
import Model, { hasMany } from '@ember-data/model';
export default class Post extends Model {
@hasMany('comment', { async: false, inverse: 'parent', as: 'commentable' }) comments;
}
Note: every Model that declares an inverse to a polymorphic relationship must declare itself exactly the same. This is because polymorphism is based on structural traits.
Polymorphic to polymorphic relationships are supported. Both sides of the relationship must be declared as polymorphic, and the as
option must be used to declare the abstract type each record satisfies on both sides.
Type Parameters
T
T
Parameters
type
TypeFromInstance
<NoNull
<T
>>
(optional) the name of the related resource
options
RelationshipOptions
<T
, boolean
>
(optional) a hash of options
Returns
RelationshipDecorator
<T
>
relationship
Call Signature
function hasMany(type, options): RelationshipDecorator<unknown>;
Defined in: packages/model/src/-private/has-many.ts:258
hasMany
is used to define Many-To-One and Many-To-Many, and Many-To-None relationships on a Model.
hasMany
takes a configuration hash as a second parameter, currently supported options are:
async
: (required) A boolean value used to declare whether this is a sync (false) or async (true) relationship.inverse
: (required) A string used to identify the inverse property on a related model, ornull
.polymorphic
: (optional) A boolean value to mark the relationship as polymorphicas
: (optional) A string used to declare the abstract type "this" record satisfies for polymorphism.
Examples
To declare a many-to-one (or one-to-many) relationship, use belongsTo
in combination with hasMany
:
// app/models/post.js
import Model, { hasMany } from '@ember-data/model';
export default class Post extends Model {
@hasMany('comment', { async: false, inverse: 'post' }) comments;
}
// app/models/comment.js
import Model, { belongsTo } from '@ember-data/model';
export default class Comment extends Model {
@belongsTo('post', { async: false, inverse: 'comments' }) post;
}
To declare a many-to-many relationship with managed inverses, use hasMany
for both sides:
// app/models/post.js
import Model, { hasMany } from '@ember-data/model';
export default class Post extends Model {
@hasMany('tag', { async: true, inverse: 'posts' }) tags;
}
// app/models/tag.js
import Model, { hasMany } from '@ember-data/model';
export default class Tag extends Model {
@hasMany('post', { async: true, inverse: 'tags' }) posts;
}
To declare a many-to-many relationship without managed inverses, use hasMany
for both sides with null
as the inverse:
// app/models/post.js
import Model, { hasMany } from '@ember-data/model';
export default class Post extends Model {
@hasMany('tag', { async: true, inverse: null }) tags;
}
// app/models/tag.js
import Model, { hasMany } from '@ember-data/model';
export default class Tag extends Model {
@hasMany('post', { async: true, inverse: null }) posts;
}
To declare a many-to-none relationship between two models, use hasMany
with inverse set to null
on just one side::
// app/models/post.js
import Model, { hasMany } from '@ember-data/model';
export default class Post extends Model {
@hasMany('category', { async: true, inverse: null }) categories;
}
Sync vs Async Relationships
EmberData fulfills relationships using resource data available in the cache.
Sync relationships point directly to the known related resources.
When a relationship is declared as async, if any of the known related resources have not been loaded, they will be fetched. The property on the record when accessed provides a promise that resolves once all resources are loaded.
Async relationships may take advantage of links. On access, if the related link has not been loaded, or if any known resources are not available in the cache, the fresh state will be fetched using the link.
In contrast to async relationship, accessing a sync relationship will error on access when any of the known related resources have not been loaded.
If you are using links
with sync relationships, you have to use the HasMany reference API to fetch or refresh related resources that aren't loaded. For instance, for a comments
relationship:
post.hasMany('comments').reload();
Polymorphic Relationships
To declare a polymorphic relationship, use hasMany
with the polymorphic
option set to true
:
// app/models/comment.js
import Model, { belongsTo } from '@ember-data/model';
export default class Comment extends Model {
@belongsTo('commentable', { async: false, inverse: 'comments', polymorphic: true }) parent;
}
'commentable'
here is referred to as the "abstract type" for the polymorphic relationship.
Polymorphic relationships with inverse: null
will accept any type of record as their content. Polymorphic relationships with inverse
set to a string will only accept records with a matching inverse relationships declaring itself as satisfying the abstract type.
Below, 'as' is used to declare the that 'post' record satisfies the abstract type 'commentable' for this relationship.
// app/models/post.js
import Model, { hasMany } from '@ember-data/model';
export default class Post extends Model {
@hasMany('comment', { async: false, inverse: 'parent', as: 'commentable' }) comments;
}
Note: every Model that declares an inverse to a polymorphic relationship must declare itself exactly the same. This is because polymorphism is based on structural traits.
Polymorphic to polymorphic relationships are supported. Both sides of the relationship must be declared as polymorphic, and the as
option must be used to declare the abstract type each record satisfies on both sides.
Parameters
type
string
(optional) the name of the related resource
options
RelationshipOptions
<unknown
, boolean
>
(optional) a hash of options
Returns
RelationshipDecorator
<unknown
>
relationship