Skip to content

Commit

Permalink
Merge pull request #35 from fizzbuds/33-implement-getbyidorthrow
Browse files Browse the repository at this point in the history
getByIdOrThrow on AggregateRepo
  • Loading branch information
lucagiove authored Apr 1, 2024
2 parents 33db0f5 + 580ef1c commit cf5e12e
Show file tree
Hide file tree
Showing 5 changed files with 64 additions and 5 deletions.
5 changes: 5 additions & 0 deletions .changeset/spotty-moles-cover.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
"@fizzbuds/ddd-toolkit": patch
---

add getByIdOrThrow to IAggregateRepo
6 changes: 6 additions & 0 deletions packages/ddd-toolkit/src/errors.ts
Original file line number Diff line number Diff line change
Expand Up @@ -15,3 +15,9 @@ export class RepoHookError extends Error {
super(message);
}
}

export class AggregateNotFoundError extends Error {
constructor(message: string) {
super(message);
}
}
18 changes: 15 additions & 3 deletions packages/ddd-toolkit/src/repo/mongo-aggregate-repo.ts
Original file line number Diff line number Diff line change
@@ -1,14 +1,15 @@
import { IRepoHooks } from './repo-hooks';
import { Collection, Document, MongoClient } from 'mongodb';
import { Collection, Document, MongoClient, WithId } from 'mongodb';
import { ISerializer } from './serializer.interface';
import { merge } from 'lodash';
import { DuplicatedIdError, OptimisticLockError, RepoHookError } from '../errors';
import { AggregateNotFoundError, DuplicatedIdError, OptimisticLockError, RepoHookError } from '../errors';
import { ILogger } from '../logger';
import { IInit } from '../init.interface';

export interface IAggregateRepo<A> {
// TODO add id as a generic type
getById: (id: string) => Promise<WithVersion<A> | null>;
getByIdOrThrow: (id: string) => Promise<WithVersion<A>>;
save: (aggregate: A) => Promise<void>;
}

Expand All @@ -23,6 +24,7 @@ const MONGODB_UNIQUE_INDEX_CONSTRAINT_ERROR = 11000;

export class MongoAggregateRepo<A, AM extends DocumentWithId> implements IAggregateRepo<A>, IInit {
protected readonly collection: Collection<AM>;

constructor(
protected readonly serializer: ISerializer<A, AM>,
protected readonly mongoClient: MongoClient,
Expand Down Expand Up @@ -94,11 +96,21 @@ export class MongoAggregateRepo<A, AM extends DocumentWithId> implements IAggreg
}
}

// TODO evaluate to implement getOrThrow
async getById(id: string): Promise<WithVersion<A> | null> {
const aggregateModel = await this.collection.findOne({ id: id } as any);
this.logger.debug(`Retrieving aggregate ${id}. Found: ${JSON.stringify(aggregateModel)}`);
if (!aggregateModel) return null;

return this.modelToAggregateWithVersion(aggregateModel);
}

async getByIdOrThrow(id: string): Promise<WithVersion<A>> {
const aggregate = await this.getById(id);
if (!aggregate) throw new AggregateNotFoundError(`Aggregate ${id} not found.`);
return aggregate;
}

private modelToAggregateWithVersion(aggregateModel: WithId<AM>): WithVersion<A> {
const aggregate = this.serializer.modelToAggregate(aggregateModel as AM);
return merge<A, { __version: number }>(aggregate, { __version: aggregateModel.__version });
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ import { MongoAggregateRepo } from '../mongo-aggregate-repo';
import { MongoMemoryReplSet } from 'mongodb-memory-server';
import { MongoClient } from 'mongodb';
import { TestAggregate, TestModel, TestSerializer } from './example.serializer';
import { AggregateNotFoundError } from '../../errors';

describe('MongoAggregateRepo MongoDB Integration', () => {
let mongodb: MongoMemoryReplSet;
Expand Down Expand Up @@ -39,7 +40,7 @@ describe('MongoAggregateRepo MongoDB Integration', () => {
});

describe('Save and Get', () => {
describe('Given an aggregate', () => {
describe('Given an existing aggregate', () => {
describe('When saving', () => {
const id1 = 'id1';
beforeEach(async () => {
Expand All @@ -54,6 +55,22 @@ describe('MongoAggregateRepo MongoDB Integration', () => {
});
});
});

describe('Given an un-existing aggregate', () => {
describe('When getById', () => {
it('should return null', async () => {
expect(await aggregateRepo.getById('not-existing-id')).toBeNull();
});
});

describe('When getByIdOrThrow', () => {
it('should throw AggregateNotFoundError', async () => {
await expect(() => aggregateRepo.getByIdOrThrow('not-existing-id')).rejects.toThrowError(
AggregateNotFoundError,
);
});
});
});
});

describe('Optimistic Lock', () => {
Expand Down
21 changes: 20 additions & 1 deletion packages/ddd-toolkit/src/repo/tests/mongo-aggregate-repo.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ import { ISerializer } from '../serializer.interface';
import { collectionMock, mongoClientMock } from './mongo.mock';

const serializerMock: ISerializer<any, any> = {
modelToAggregate: jest.fn(),
modelToAggregate: jest.fn().mockReturnValue({ id: 'id' }),
aggregateToModel: jest.fn().mockReturnValue({ id: 'id' }),
};

Expand Down Expand Up @@ -34,6 +34,25 @@ describe('MongoAggregateRepo', () => {
});
});

describe('getByIdOrThrow', () => {
it('should call findOne with id', async () => {
try {
await mongoAggregateRepo.getByIdOrThrow('id');
} catch (e) {}
expect(collectionMock.findOne).toHaveBeenCalledWith({ id: 'id' });
});

it('should throw AggregateNotFoundError if not found', async () => {
collectionMock.findOne.mockResolvedValue(null);
await expect(mongoAggregateRepo.getByIdOrThrow('id')).rejects.toThrow();
});

it('should return aggregate if found', async () => {
collectionMock.findOne.mockResolvedValue({ id: 'id' });
expect(await mongoAggregateRepo.getByIdOrThrow('id')).toMatchObject({ id: 'id' });
});
});

describe('save', () => {
it('should call updateOne with id', async () => {
await mongoAggregateRepo.save({ id: 'id' });
Expand Down

0 comments on commit cf5e12e

Please sign in to comment.