Skip to content

Commit

Permalink
test(repository): add tests for hasManyThrough helpers
Browse files Browse the repository at this point in the history
  • Loading branch information
Agnes Lin authored and agnes512 committed May 15, 2020
1 parent fe4cf5e commit 1168c5f
Showing 1 changed file with 327 additions and 0 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,327 @@
// Copyright IBM Corp. 2019. All Rights Reserved.
// Node module: @loopback/repository
// This file is licensed under the MIT License.
// License text available at https://opensource.org/licenses/MIT

import {expect} from '@loopback/testlab';
import {
Entity,
HasManyDefinition,
model,
ModelDefinition,
property,
RelationType,
} from '../../../..';
import {
createTargetConstraint,
createThroughConstraint,
HasManyThroughResolvedDefinition,
resolveHasManyThroughMetadata,
} from '../../../../relations/has-many/has-many-through.helpers';

describe('HasManyThroughHelpers', () => {
context('createThroughConstraint', () => {
it('can create constraint for searching through models', () => {
const resolved = resolvedMetadata as HasManyThroughResolvedDefinition;
const result = createThroughConstraint(resolved, 1);
expect(result).to.containEql({categoryId: 1});
});
});
context('createTargetConstraint', () => {
it('can create constraint for searching target models', () => {
const through1 = createCategoryProductLink({
id: 1,
categoryId: 2,
productId: 9,
});
const through2 = createCategoryProductLink({
id: 2,
categoryId: 2,
productId: 8,
});
const resolved = resolvedMetadata as HasManyThroughResolvedDefinition;

// single through model
let result = createTargetConstraint(resolved, [through1]);
expect(result).to.containEql({id: 9});
// multiple through models
result = createTargetConstraint(resolved, [through1, through2]);
expect(result).to.containEql({id: {inq: [9, 8]}});
});
});
context('resolveHasManyThroughMetadata', () => {
it('throws if the wrong metadata type is used', async () => {
const metadata: unknown = {
name: 'category',
type: RelationType.hasOne,
targetsMany: false,
source: Category,
target: () => Product,
};

expect(() => {
resolveHasManyThroughMetadata(metadata as HasManyDefinition);
}).to.throw(
/Invalid hasOne definition for Category#category: relation type must be HasMany/,
);
});

it('throws if the through is not provided', async () => {
const metadata: unknown = {
name: 'category',
type: RelationType.hasMany,
targetsMany: true,
source: Category,
target: () => Product,
};

expect(() => {
resolveHasManyThroughMetadata(metadata as HasManyDefinition);
}).to.throw(
/Invalid hasMany definition for Category#category: through must be specified/,
);
});

it('throws if the through is not resolvable', async () => {
const metadata: unknown = {
name: 'category',
type: RelationType.hasMany,
targetsMany: true,
source: Category,
through: {model: 'random'},
target: () => Product,
};

expect(() => {
resolveHasManyThroughMetadata(metadata as HasManyDefinition);
}).to.throw(
/Invalid hasMany definition for Category#category: through.model must be a type resolver/,
);
});

describe('resolves through.keyTo/keyFrom', () => {
it('resolves metadata with complete hasManyThrough definition', () => {
const metadata = {
name: 'products',
type: RelationType.hasMany,
targetsMany: true,
source: Category,
keyFrom: 'id',
target: () => Product,
keyTo: 'id',

through: {
model: () => CategoryProductLink,
keyFrom: 'categoryId',
keyTo: 'productId',
},
};
const meta = resolveHasManyThroughMetadata(
metadata as HasManyDefinition,
);

expect(meta).to.eql(resolvedMetadata);
});

it('infers through.keyFrom if it is not provided', () => {
const metadata = {
name: 'products',
type: RelationType.hasMany,
targetsMany: true,
source: Category,
keyFrom: 'id',
target: () => Product,
keyTo: 'id',

through: {
model: () => CategoryProductLink,
// no through.keyFrom
keyTo: 'productId',
},
};
const meta = resolveHasManyThroughMetadata(
metadata as HasManyDefinition,
);

expect(meta).to.eql(resolvedMetadata);
});

it('infers through.keyTo if it is not provided', () => {
const metadata = {
name: 'products',
type: RelationType.hasMany,
targetsMany: true,
source: Category,
keyFrom: 'id',
target: () => Product,
keyTo: 'id',

through: {
model: () => CategoryProductLink,
keyFrom: 'categoryId',
// no through.keyTo
},
};

const meta = resolveHasManyThroughMetadata(
metadata as HasManyDefinition,
);

expect(meta).to.eql(resolvedMetadata);
});

it('throws if through.keyFrom is not provided in through', async () => {
const metadata = {
name: 'categories',
type: RelationType.hasMany,
targetsMany: true,
source: Category,
keyFrom: 'id',
target: () => Product,
keyTo: 'id',

through: {
model: () => InvalidThrough,
keyTo: 'productId',
},
};

expect(() => {
resolveHasManyThroughMetadata(metadata as HasManyDefinition);
}).to.throw(
/Invalid hasMany definition for Category#categories: through model InvalidThrough is missing definition of source foreign key/,
);
});

it('throws if through.keyTo is not provided in through', async () => {
const metadata = {
name: 'categories',
type: RelationType.hasMany,
targetsMany: true,
source: Category,
keyFrom: 'id',
target: () => Product,
keyTo: 'id',

through: {
model: () => InvalidThrough2,
keyFrom: 'categoryId',
},
};

expect(() => {
resolveHasManyThroughMetadata(metadata as HasManyDefinition);
}).to.throw(
/Invalid hasMany definition for Category#categories: through model InvalidThrough2 is missing definition of target foreign key/,
);
});

it('throws if the tarhet model does not have the id property', async () => {
const metadata = {
name: 'categories',
type: RelationType.hasMany,
targetsMany: true,
source: Category,
keyFrom: 'id',
target: () => InvalidProduct,
keyTo: 'id',

through: {
model: () => CategoryProductLink,
keyFrom: 'categoryId',
keyTo: 'productId',
},
};

expect(() => {
resolveHasManyThroughMetadata(metadata as HasManyDefinition);
}).to.throw(
'Invalid hasMany definition for Category#categories: target model InvalidProduct does not have any primary key (id property)',
);
});
});
});
/****** HELPERS *******/

@model()
class Category extends Entity {
@property({id: true})
id: number;

constructor(data: Partial<Category>) {
super(data);
}
}

@model()
class Product extends Entity {
@property({id: true})
id: number;

constructor(data: Partial<Product>) {
super(data);
}
}

@model()
class InvalidProduct extends Entity {
@property({id: false})
random: number;
constructor(data: Partial<InvalidProduct>) {
super(data);
}
}

@model()
class CategoryProductLink extends Entity {
@property({id: true})
id: number;
@property()
categoryId: number;
@property()
productId: number;

constructor(data: Partial<Product>) {
super(data);
}
}

const resolvedMetadata = {
name: 'products',
type: 'hasMany',
targetsMany: true,
source: Category,
keyFrom: 'id',
target: () => Product,
keyTo: 'id',
through: {
model: () => CategoryProductLink,
keyFrom: 'categoryId',
keyTo: 'productId',
},
};

class InvalidThrough extends Entity {}
InvalidThrough.definition = new ModelDefinition('InvalidThrough')
.addProperty('id', {
type: 'number',
id: true,
required: true,
})
// lack through.keyFrom
.addProperty('productId', {type: 'number'});

class InvalidThrough2 extends Entity {}
InvalidThrough2.definition = new ModelDefinition('InvalidThrough2')
.addProperty('id', {
type: 'number',
id: true,
required: true,
})
// lack through.keyTo
.addProperty('categoryId', {type: 'number'});

function createCategoryProductLink(properties: Partial<CategoryProductLink>) {
return new CategoryProductLink(properties);
}
});

0 comments on commit 1168c5f

Please sign in to comment.