You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
175 lines
5.6 KiB
175 lines
5.6 KiB
3 years ago
|
import type { Document } from '../bson';
|
||
|
import type { Collection } from '../collection';
|
||
|
import { MongoCompatibilityError, MongoServerError } from '../error';
|
||
|
import type { Server } from '../sdam/server';
|
||
|
import type { ClientSession } from '../sessions';
|
||
|
import type { Callback, MongoDBNamespace } from '../utils';
|
||
|
import type { WriteConcernOptions } from '../write_concern';
|
||
|
import { CollationOptions, CommandOperation, CommandOperationOptions } from './command';
|
||
|
import { Aspect, defineAspects, Hint } from './operation';
|
||
|
|
||
|
/** @public */
|
||
|
export interface DeleteOptions extends CommandOperationOptions, WriteConcernOptions {
|
||
|
/** If true, when an insert fails, don't execute the remaining writes. If false, continue with remaining inserts when one fails. */
|
||
|
ordered?: boolean;
|
||
|
/** Specifies the collation to use for the operation */
|
||
|
collation?: CollationOptions;
|
||
|
/** Specify that the update query should only consider plans using the hinted index */
|
||
|
hint?: string | Document;
|
||
|
/** Map of parameter names and values that can be accessed using $$var (requires MongoDB 5.0). */
|
||
|
let?: Document;
|
||
|
}
|
||
|
|
||
|
/** @public */
|
||
|
export interface DeleteResult {
|
||
|
/** Indicates whether this write result was acknowledged. If not, then all other members of this result will be undefined. */
|
||
|
acknowledged: boolean;
|
||
|
/** The number of documents that were deleted */
|
||
|
deletedCount: number;
|
||
|
}
|
||
|
|
||
|
/** @public */
|
||
|
export interface DeleteStatement {
|
||
|
/** The query that matches documents to delete. */
|
||
|
q: Document;
|
||
|
/** The number of matching documents to delete. */
|
||
|
limit: number;
|
||
|
/** Specifies the collation to use for the operation. */
|
||
|
collation?: CollationOptions;
|
||
|
/** A document or string that specifies the index to use to support the query predicate. */
|
||
|
hint?: Hint;
|
||
|
}
|
||
|
|
||
|
/** @internal */
|
||
|
export class DeleteOperation extends CommandOperation<DeleteResult> {
|
||
|
override options: DeleteOptions;
|
||
|
statements: DeleteStatement[];
|
||
|
|
||
|
constructor(ns: MongoDBNamespace, statements: DeleteStatement[], options: DeleteOptions) {
|
||
|
super(undefined, options);
|
||
|
this.options = options;
|
||
|
this.ns = ns;
|
||
|
this.statements = statements;
|
||
|
}
|
||
|
|
||
|
override get canRetryWrite(): boolean {
|
||
|
if (super.canRetryWrite === false) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
return this.statements.every(op => (op.limit != null ? op.limit > 0 : true));
|
||
|
}
|
||
|
|
||
|
override execute(server: Server, session: ClientSession | undefined, callback: Callback): void {
|
||
|
const options = this.options ?? {};
|
||
|
const ordered = typeof options.ordered === 'boolean' ? options.ordered : true;
|
||
|
const command: Document = {
|
||
|
delete: this.ns.collection,
|
||
|
deletes: this.statements,
|
||
|
ordered
|
||
|
};
|
||
|
|
||
|
if (options.let) {
|
||
|
command.let = options.let;
|
||
|
}
|
||
|
|
||
|
// we check for undefined specifically here to allow falsy values
|
||
|
// eslint-disable-next-line no-restricted-syntax
|
||
|
if (options.comment !== undefined) {
|
||
|
command.comment = options.comment;
|
||
|
}
|
||
|
|
||
|
const unacknowledgedWrite = this.writeConcern && this.writeConcern.w === 0;
|
||
|
if (unacknowledgedWrite) {
|
||
|
if (this.statements.find((o: Document) => o.hint)) {
|
||
|
// TODO(NODE-3541): fix error for hint with unacknowledged writes
|
||
|
callback(new MongoCompatibilityError(`hint is not supported with unacknowledged writes`));
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
super.executeCommand(server, session, command, callback);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
export class DeleteOneOperation extends DeleteOperation {
|
||
|
constructor(collection: Collection, filter: Document, options: DeleteOptions) {
|
||
|
super(collection.s.namespace, [makeDeleteStatement(filter, { ...options, limit: 1 })], options);
|
||
|
}
|
||
|
|
||
|
override execute(
|
||
|
server: Server,
|
||
|
session: ClientSession | undefined,
|
||
|
callback: Callback<DeleteResult>
|
||
|
): void {
|
||
|
super.execute(server, session, (err, res) => {
|
||
|
if (err || res == null) return callback(err);
|
||
|
if (res.code) return callback(new MongoServerError(res));
|
||
|
if (res.writeErrors) return callback(new MongoServerError(res.writeErrors[0]));
|
||
|
if (this.explain) return callback(undefined, res);
|
||
|
|
||
|
callback(undefined, {
|
||
|
acknowledged: this.writeConcern?.w !== 0 ?? true,
|
||
|
deletedCount: res.n
|
||
|
});
|
||
|
});
|
||
|
}
|
||
|
}
|
||
|
|
||
|
export class DeleteManyOperation extends DeleteOperation {
|
||
|
constructor(collection: Collection, filter: Document, options: DeleteOptions) {
|
||
|
super(collection.s.namespace, [makeDeleteStatement(filter, options)], options);
|
||
|
}
|
||
|
|
||
|
override execute(
|
||
|
server: Server,
|
||
|
session: ClientSession | undefined,
|
||
|
callback: Callback<DeleteResult>
|
||
|
): void {
|
||
|
super.execute(server, session, (err, res) => {
|
||
|
if (err || res == null) return callback(err);
|
||
|
if (res.code) return callback(new MongoServerError(res));
|
||
|
if (res.writeErrors) return callback(new MongoServerError(res.writeErrors[0]));
|
||
|
if (this.explain) return callback(undefined, res);
|
||
|
|
||
|
callback(undefined, {
|
||
|
acknowledged: this.writeConcern?.w !== 0 ?? true,
|
||
|
deletedCount: res.n
|
||
|
});
|
||
|
});
|
||
|
}
|
||
|
}
|
||
|
|
||
|
export function makeDeleteStatement(
|
||
|
filter: Document,
|
||
|
options: DeleteOptions & { limit?: number }
|
||
|
): DeleteStatement {
|
||
|
const op: DeleteStatement = {
|
||
|
q: filter,
|
||
|
limit: typeof options.limit === 'number' ? options.limit : 0
|
||
|
};
|
||
|
|
||
|
if (options.collation) {
|
||
|
op.collation = options.collation;
|
||
|
}
|
||
|
|
||
|
if (options.hint) {
|
||
|
op.hint = options.hint;
|
||
|
}
|
||
|
|
||
|
return op;
|
||
|
}
|
||
|
|
||
|
defineAspects(DeleteOperation, [Aspect.RETRYABLE, Aspect.WRITE_OPERATION]);
|
||
|
defineAspects(DeleteOneOperation, [
|
||
|
Aspect.RETRYABLE,
|
||
|
Aspect.WRITE_OPERATION,
|
||
|
Aspect.EXPLAINABLE,
|
||
|
Aspect.SKIP_COLLATION
|
||
|
]);
|
||
|
defineAspects(DeleteManyOperation, [
|
||
|
Aspect.WRITE_OPERATION,
|
||
|
Aspect.EXPLAINABLE,
|
||
|
Aspect.SKIP_COLLATION
|
||
|
]);
|