Skip to content

Commit

Permalink
fix: Fix Mongodb delete by ObjectId. Closes #6552 (#6553)
Browse files Browse the repository at this point in the history
* fix: Fix Mongodb delete by ObjectId. Closes #6552

* add regression test
  • Loading branch information
danmana committed Oct 7, 2020
1 parent bc60dd5 commit e37eb1e
Show file tree
Hide file tree
Showing 4 changed files with 165 additions and 3 deletions.
19 changes: 16 additions & 3 deletions src/entity-manager/MongoEntityManager.ts
Original file line number Diff line number Diff line change
Expand Up @@ -630,6 +630,18 @@ export class MongoEntityManager extends EntityManager {
* Ensures given id is an id for query.
*/
protected convertMixedCriteria(metadata: EntityMetadata, idMap: any): ObjectLiteral {
const objectIdInstance = PlatformTools.load("mongodb").ObjectID;

// check first if it's ObjectId compatible:
// string, number, Buffer, ObjectId or ObjectId-like
if (objectIdInstance.isValid(idMap)) {
return {
"_id": new objectIdInstance(idMap)
};
}

// if it's some other type of object build a query from the columns
// this check needs to be after the ObjectId check, because a valid ObjectId is also an Object instance
if (idMap instanceof Object) {
return metadata.columns.reduce((query, column) => {
const columnValue = column.getEntityValue(idMap);
Expand All @@ -639,10 +651,11 @@ export class MongoEntityManager extends EntityManager {
}, {} as any);
}

// means idMap is just object id
const objectIdInstance = PlatformTools.load("mongodb").ObjectID;
// last resort: try to convert it to an ObjectID anyway
// most likely it will fail, but we want to be backwards compatible and keep the same thrown Errors.
// it can still pass with null/undefined
return {
"_id": (idMap instanceof objectIdInstance) ? idMap : new objectIdInstance(idMap)
"_id": new objectIdInstance(idMap)
};
}

Expand Down
14 changes: 14 additions & 0 deletions test/github-issues/6552/entity/Post.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
import { ObjectID, ObjectIdColumn } from "../../../../src";
import { Column } from "../../../../src/decorator/columns/Column";
import { Entity } from "../../../../src/decorator/entity/Entity";

@Entity()
export class Post {

@ObjectIdColumn()
_id: ObjectID;

@Column()
title: string;

}
14 changes: 14 additions & 0 deletions test/github-issues/6552/entity/PostV2.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
import { ObjectID, ObjectIdColumn } from "../../../../src";
import { Column } from "../../../../src/decorator/columns/Column";
import { Entity } from "../../../../src/decorator/entity/Entity";

@Entity()
export class PostV2 {

@ObjectIdColumn()
postId: ObjectID;

@Column()
title: string;

}
121 changes: 121 additions & 0 deletions test/github-issues/6552/issue-6552.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,121 @@
import { expect } from "chai";
import "reflect-metadata";
import { Connection } from "../../../src/connection/Connection";
import { PlatformTools } from "../../../src/platform/PlatformTools";
import { closeTestingConnections, createTestingConnections, reloadTestingDatabases } from "../../utils/test-utils";
import { Post } from "./entity/Post";
import { PostV2 } from "./entity/PostV2";
import { FindConditions } from "../../../src";

describe("github issues > #6552 MongoRepository delete by ObjectId deletes the wrong entity", () => {

let connections: Connection[];
before(async () => connections = await createTestingConnections({
entities: [__dirname + "/entity/*{.js,.ts}"],
enabledDrivers: ["mongodb"]
}));
beforeEach(() => reloadTestingDatabases(connections));
after(() => closeTestingConnections(connections));

// before the fix this would delete incorrectly post1 instead of post2
it("should delete the correct entity when id column is called _id", () => Promise.all(connections.map(async function (connection) {

// setup: create 2 posts
const post1 = new Post();
post1.title = "Post 1";
await connection.manager.save(post1);

const post2 = new Post();
post2.title = "Post 2";
await connection.manager.save(post2);

const objectIdInstance = PlatformTools.load("mongodb").ObjectID;

// double check that post2._id is actually an ObjectID
expect(post2._id).to.be.not.null;
expect(post2._id).to.be.not.undefined;
expect(post2._id).to.be.instanceof(objectIdInstance);

// delete Post 2 by ObjectId directly
await connection.manager.delete(Post, post2._id);
// This used to wrongly perform deleteOne({}) - deleting the first Post in the collection

// Post 1 should remain in the DB
const count1 = await connection.manager.count(Post, { _id: post1._id } as FindConditions<Post>);
expect(count1).to.be.equal(1, "Post 1 should still exist");

// Post 2 should be deleted
const count2 = await connection.manager.count(Post, { _id: post2._id } as FindConditions<Post>);
expect(count2).to.be.equal(0, "Post 2 should be deleted");


})));

// before the fix this wouldn't delete anything
it("should delete the correct entity when id column is not called _id", () => Promise.all(connections.map(async function (connection) {

// setup: create 2 posts
const post1 = new PostV2();
post1.title = "Post 1";
await connection.manager.save(post1);

const post2 = new PostV2();
post2.title = "Post 2";
await connection.manager.save(post2);

const objectIdInstance = PlatformTools.load("mongodb").ObjectID;

// double check that post2.postId is actually an ObjectID
expect(post2.postId).to.be.not.null;
expect(post2.postId).to.be.not.undefined;
expect(post2.postId).to.be.instanceof(objectIdInstance);

// delete Post 2 by ObjectId directly
await connection.manager.delete(PostV2, post2.postId);
// This used to wrongly perform deleteOne({_id: Buffer}) - not deleting anything because Buffer is not an ObjectId

// Post 1 should remain in the DB
const count1 = await connection.manager.count(PostV2, { _id: post1.postId } as FindConditions<PostV2>);
expect(count1).to.be.equal(1, "Post 1 should still exist");

// Post 2 should be deleted
const count2 = await connection.manager.count(PostV2, { _id: post2.postId } as FindConditions<PostV2>);
expect(count2).to.be.equal(0, "Post 2 should be deleted");


})));

// before the fix this passed (added here to make sure we don't cause any regressions)
it("should delete the correct entity when deleting by _id query", () => Promise.all(connections.map(async function (connection) {

// setup: create 2 posts
const post1 = new Post();
post1.title = "Post 1";
await connection.manager.save(post1);

const post2 = new Post();
post2.title = "Post 2";
await connection.manager.save(post2);

const objectIdInstance = PlatformTools.load("mongodb").ObjectID;

// double check that post2._id is actually an ObjectID
expect(post2._id).to.be.not.null;
expect(post2._id).to.be.not.undefined;
expect(post2._id).to.be.instanceof(objectIdInstance);

// delete Post 2 by ObjectId directly
await connection.manager.delete(Post, { _id: post2._id });

// Post 1 should remain in the DB
const count1 = await connection.manager.count(Post, { _id: post1._id } as FindConditions<Post>);
expect(count1).to.be.equal(1, "Post 1 should still exist");

// Post 2 should be deleted
const count2 = await connection.manager.count(Post, { _id: post2._id } as FindConditions<Post>);
expect(count2).to.be.equal(0, "Post 2 should be deleted");


})));

});

0 comments on commit e37eb1e

Please sign in to comment.