-
Notifications
You must be signed in to change notification settings - Fork 1.1k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
test(repository-tests): adding acceptance tests for hasManyThrough re…
…lation
- Loading branch information
Showing
12 changed files
with
449 additions
and
12 deletions.
There are no files selected for viewing
248 changes: 248 additions & 0 deletions
248
...es/repository-tests/src/crud/relations/acceptance/has-many-through.relation.acceptance.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,248 @@ | ||
// Copyright IBM Corp. 2020. All Rights Reserved. | ||
// Node module: @loopback/repository-tests | ||
// This file is licensed under the MIT License. | ||
// License text available at https://opensource.org/licenses/MIT | ||
|
||
import {expect, toJSON} from '@loopback/testlab'; | ||
import { | ||
CrudFeatures, | ||
CrudRepositoryCtor, | ||
CrudTestContext, | ||
DataSourceOptions, | ||
} from '../../..'; | ||
import { | ||
deleteAllModelsInDefaultDataSource, | ||
MixedIdType, | ||
withCrudCtx, | ||
} from '../../../helpers.repository-tests'; | ||
import { | ||
CartItem, | ||
CartItemRepository, | ||
Customer, | ||
CustomerCartItemLink, | ||
CustomerCartItemLinkRepository, | ||
CustomerRepository, | ||
} from '../fixtures/models'; | ||
import {givenBoundCrudRepositories} from '../helpers'; | ||
|
||
export function hasManyThroughRelationAcceptance( | ||
dataSourceOptions: DataSourceOptions, | ||
repositoryClass: CrudRepositoryCtor, | ||
features: CrudFeatures, | ||
) { | ||
describe('HasManyThrough relation (acceptance)', () => { | ||
before(deleteAllModelsInDefaultDataSource); | ||
let customerRepo: CustomerRepository; | ||
let cartItemRepo: CartItemRepository; | ||
let customerCartItemLinkRepo: CustomerCartItemLinkRepository; | ||
let existingCustomerId: MixedIdType; | ||
|
||
before( | ||
withCrudCtx(async function setupRepository(ctx: CrudTestContext) { | ||
({ | ||
customerRepo, | ||
cartItemRepo, | ||
customerCartItemLinkRepo, | ||
} = givenBoundCrudRepositories( | ||
ctx.dataSource, | ||
repositoryClass, | ||
features, | ||
)); | ||
await ctx.dataSource.automigrate([ | ||
Customer.name, | ||
CartItem.name, | ||
CustomerCartItemLink.name, | ||
]); | ||
}), | ||
); | ||
|
||
beforeEach(async () => { | ||
await customerRepo.deleteAll(); | ||
await cartItemRepo.deleteAll(); | ||
await customerCartItemLinkRepo.deleteAll(); | ||
}); | ||
|
||
beforeEach(async () => { | ||
existingCustomerId = (await givenPersistedCustomerInstance()).id; | ||
}); | ||
|
||
it('can create an instance of the related model alone with a through model', async () => { | ||
const item = await customerRepo | ||
.cartItems(existingCustomerId) | ||
.create( | ||
{description: 'an item'}, | ||
{throughData: {description: 'a through model'}}, | ||
); | ||
|
||
expect(toJSON(item)).containDeep( | ||
toJSON({ | ||
id: item.id, | ||
description: 'an item', | ||
}), | ||
); | ||
|
||
const persistedItem = await cartItemRepo.findById(item.id); | ||
expect(toJSON(persistedItem)).to.deepEqual(toJSON(item)); | ||
const persistedLink = await customerCartItemLinkRepo.find(); | ||
expect(toJSON(persistedLink[0])).to.containDeep( | ||
toJSON({ | ||
customerId: existingCustomerId, | ||
cartItemId: item.id, | ||
description: 'a through model', | ||
}), | ||
); | ||
}); | ||
|
||
it('can find instances of the related model', async () => { | ||
const item = await customerRepo | ||
.cartItems(existingCustomerId) | ||
.create( | ||
{description: 'an item'}, | ||
{throughData: {description: 'a through model'}}, | ||
); | ||
const notMyItem = await cartItemRepo.create({ | ||
description: "someone else's item", | ||
}); | ||
const result = await customerRepo.cartItems(existingCustomerId).find(); | ||
expect(toJSON(result)).to.containDeep(toJSON([item])); | ||
expect(toJSON(result[0])).to.not.containEql(toJSON(notMyItem)); | ||
}); | ||
|
||
it('can patch instances', async () => { | ||
const item1 = await customerRepo | ||
.cartItems(existingCustomerId) | ||
.create({description: 'group 1'}); | ||
const item2 = await customerRepo | ||
.cartItems(existingCustomerId) | ||
.create({description: 'group 1'}); | ||
const count = await customerRepo | ||
.cartItems(existingCustomerId) | ||
.patch({description: 'updated'}); | ||
|
||
expect(count.count).to.equal(2); | ||
const result = await customerRepo.cartItems(existingCustomerId).find(); | ||
expect(toJSON(result)).to.containDeep( | ||
toJSON([ | ||
{id: item1.id, description: 'updated'}, | ||
{id: item2.id, description: 'updated'}, | ||
]), | ||
); | ||
}); | ||
|
||
it('can patch an instance based on the filter', async () => { | ||
const item1 = await customerRepo | ||
.cartItems(existingCustomerId) | ||
.create({description: 'group 1'}); | ||
const item2 = await customerRepo | ||
.cartItems(existingCustomerId) | ||
.create({description: 'group 1'}); | ||
const count = await customerRepo | ||
.cartItems(existingCustomerId) | ||
.patch({description: 'group 2'}, {id: item2.id}); | ||
|
||
expect(count.count).to.equal(1); | ||
const result = await customerRepo.cartItems(existingCustomerId).find(); | ||
expect(toJSON(result)).to.containDeep( | ||
toJSON([ | ||
{id: item1.id, description: 'group 1'}, | ||
{id: item2.id, description: 'group 2'}, | ||
]), | ||
); | ||
}); | ||
|
||
it('throws error when query tries to change the target id', async () => { | ||
// a diff id for CartItem instance | ||
const anotherId = (await givenPersistedCustomerInstance()).id; | ||
await customerRepo | ||
.cartItems(existingCustomerId) | ||
.create({description: 'group 1'}); | ||
|
||
await expect( | ||
customerRepo.cartItems(existingCustomerId).patch({id: anotherId}), | ||
).to.be.rejectedWith(/Property "id" cannot be changed!/); | ||
}); | ||
|
||
it('can delete many instances and their through models', async () => { | ||
await customerRepo | ||
.cartItems(existingCustomerId) | ||
.create({description: 'group 1'}); | ||
await customerRepo | ||
.cartItems(existingCustomerId) | ||
.create({description: 'group 1'}); | ||
|
||
let links = await customerCartItemLinkRepo.find(); | ||
let cartItems = await cartItemRepo.find(); | ||
expect(links).have.length(2); | ||
expect(cartItems).have.length(2); | ||
|
||
await customerRepo.cartItems(existingCustomerId).delete(); | ||
links = await customerCartItemLinkRepo.find(); | ||
cartItems = await cartItemRepo.find(); | ||
expect(links).have.length(0); | ||
expect(cartItems).have.length(0); | ||
}); | ||
|
||
it('can delete corresponding through models when the target gets deleted', async () => { | ||
const item = await customerRepo | ||
.cartItems(existingCustomerId) | ||
.create({description: 'group 1'}); | ||
const anotherId = (await givenPersistedCustomerInstance()).id; | ||
// another through model that links to the same item | ||
await customerCartItemLinkRepo.create({ | ||
customerId: anotherId, | ||
cartItemId: item.id, | ||
}); | ||
|
||
let links = await customerCartItemLinkRepo.find(); | ||
let cartItems = await cartItemRepo.find(); | ||
expect(links).have.length(2); | ||
expect(cartItems).have.length(1); | ||
|
||
await customerRepo.cartItems(existingCustomerId).delete(); | ||
links = await customerCartItemLinkRepo.find(); | ||
cartItems = await cartItemRepo.find(); | ||
expect(links).have.length(0); | ||
expect(cartItems).have.length(0); | ||
}); | ||
|
||
//FIXME(Agnes): should be able to do deletion based on filters | ||
|
||
it('can link a target model to a source model', async () => { | ||
const item = await cartItemRepo.create({description: 'an item'}); | ||
|
||
let targets = await customerRepo.cartItems(existingCustomerId).find(); | ||
expect(targets).to.be.empty(); | ||
|
||
await customerRepo.cartItems(existingCustomerId).link(item.id); | ||
|
||
targets = await customerRepo.cartItems(existingCustomerId).find(); | ||
expect(targets).to.deepEqual([item]); | ||
|
||
const link = await customerCartItemLinkRepo.find(); | ||
expect(toJSON(link[0])).to.containEql( | ||
toJSON({customerId: existingCustomerId, cartItemId: item.id}), | ||
); | ||
}); | ||
|
||
it('can unlink a target model from a source model', async () => { | ||
const item = await customerRepo | ||
.cartItems(existingCustomerId) | ||
.create({description: 'an item'}); | ||
|
||
let targets = await customerRepo.cartItems(existingCustomerId).find(); | ||
expect(targets).to.deepEqual([item]); | ||
|
||
await customerRepo.cartItems(existingCustomerId).unlink(item.id); | ||
|
||
targets = await customerRepo.cartItems(existingCustomerId).find(); | ||
expect(targets).to.be.empty(); | ||
|
||
const link = await customerCartItemLinkRepo.find(); | ||
expect(link).to.be.empty(); | ||
}); | ||
|
||
async function givenPersistedCustomerInstance() { | ||
return customerRepo.create({name: 'a customer'}); | ||
} | ||
}); | ||
} |
34 changes: 34 additions & 0 deletions
34
packages/repository-tests/src/crud/relations/fixtures/models/cart-item.model.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,34 @@ | ||
// Copyright IBM Corp. 2020. All Rights Reserved. | ||
// Node module: @loopback/repository-tests | ||
// This file is licensed under the MIT License. | ||
// License text available at https://opensource.org/licenses/MIT | ||
|
||
import { | ||
Entity, | ||
EntityCrudRepository, | ||
model, | ||
property, | ||
} from '@loopback/repository'; | ||
import {MixedIdType} from '../../../../helpers.repository-tests'; | ||
|
||
@model() | ||
export class CartItem extends Entity { | ||
@property({ | ||
id: true, | ||
generated: true, | ||
useDefaultIdType: true, | ||
}) | ||
id: MixedIdType; | ||
|
||
@property({ | ||
type: 'string', | ||
}) | ||
description: string; | ||
} | ||
|
||
export interface CartItemRelations {} | ||
|
||
export type CartItemWithRelations = CartItem & CartItemRelations; | ||
|
||
export interface CartItemRepository | ||
extends EntityCrudRepository<CartItem, typeof CartItem.prototype.id> {} |
44 changes: 44 additions & 0 deletions
44
...ages/repository-tests/src/crud/relations/fixtures/models/customer-cart-item-link.model.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,44 @@ | ||
// Copyright IBM Corp. 2020. All Rights Reserved. | ||
// Node module: @loopback/repository-tests | ||
// This file is licensed under the MIT License. | ||
// License text available at https://opensource.org/licenses/MIT | ||
|
||
import { | ||
Entity, | ||
EntityCrudRepository, | ||
model, | ||
property, | ||
} from '@loopback/repository'; | ||
import {MixedIdType} from '../../../../helpers.repository-tests'; | ||
|
||
@model() | ||
export class CustomerCartItemLink extends Entity { | ||
@property({ | ||
id: true, | ||
generated: true, | ||
useDefaultIdType: true, | ||
}) | ||
id: MixedIdType; | ||
|
||
@property() | ||
customerId: MixedIdType; | ||
|
||
@property() | ||
cartItemId: MixedIdType; | ||
|
||
@property({ | ||
type: 'string', | ||
}) | ||
description?: string; | ||
} | ||
|
||
export interface CustomerCartItemLinkRelations {} | ||
|
||
export type CustomerCartItemLinkWithRelations = CustomerCartItemLink & | ||
CustomerCartItemLinkRelations; | ||
|
||
export interface CustomerCartItemLinkRepository | ||
extends EntityCrudRepository< | ||
CustomerCartItemLink, | ||
typeof CustomerCartItemLink.prototype.id | ||
> {} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.