Source: lib/bulk/common.js

  1. 'use strict';
  2. var Long = require('mongodb-core').BSON.Long,
  3. MongoError = require('mongodb-core').MongoError,
  4. util = require('util');
  5. // Error codes
  6. var UNKNOWN_ERROR = 8;
  7. var INVALID_BSON_ERROR = 22;
  8. var WRITE_CONCERN_ERROR = 64;
  9. var MULTIPLE_ERROR = 65;
  10. // Insert types
  11. var INSERT = 1;
  12. var UPDATE = 2;
  13. var REMOVE = 3;
  14. /**
  15. * Helper function to define properties
  16. * @ignore
  17. */
  18. var defineReadOnlyProperty = function(self, name, value) {
  19. Object.defineProperty(self, name, {
  20. enumerable: true,
  21. get: function() {
  22. return value;
  23. }
  24. });
  25. };
  26. /**
  27. * Keeps the state of a unordered batch so we can rewrite the results
  28. * correctly after command execution
  29. * @ignore
  30. */
  31. var Batch = function(batchType, originalZeroIndex) {
  32. this.originalZeroIndex = originalZeroIndex;
  33. this.currentIndex = 0;
  34. this.originalIndexes = [];
  35. this.batchType = batchType;
  36. this.operations = [];
  37. this.size = 0;
  38. this.sizeBytes = 0;
  39. };
  40. /**
  41. * Wraps a legacy operation so we can correctly rewrite it's error
  42. * @ignore
  43. */
  44. var LegacyOp = function(batchType, operation, index) {
  45. this.batchType = batchType;
  46. this.index = index;
  47. this.operation = operation;
  48. };
  49. /**
  50. * Create a new BulkWriteResult instance (INTERNAL TYPE, do not instantiate directly)
  51. *
  52. * @class
  53. * @property {boolean} ok Did bulk operation correctly execute
  54. * @property {number} nInserted number of inserted documents
  55. * @property {number} nUpdated number of documents updated logically
  56. * @property {number} nUpserted Number of upserted documents
  57. * @property {number} nModified Number of documents updated physically on disk
  58. * @property {number} nRemoved Number of removed documents
  59. * @return {BulkWriteResult} a BulkWriteResult instance
  60. */
  61. var BulkWriteResult = function(bulkResult) {
  62. defineReadOnlyProperty(this, 'ok', bulkResult.ok);
  63. defineReadOnlyProperty(this, 'nInserted', bulkResult.nInserted);
  64. defineReadOnlyProperty(this, 'nUpserted', bulkResult.nUpserted);
  65. defineReadOnlyProperty(this, 'nMatched', bulkResult.nMatched);
  66. defineReadOnlyProperty(this, 'nModified', bulkResult.nModified);
  67. defineReadOnlyProperty(this, 'nRemoved', bulkResult.nRemoved);
  68. /**
  69. * Return an array of inserted ids
  70. *
  71. * @return {object[]}
  72. */
  73. this.getInsertedIds = function() {
  74. return bulkResult.insertedIds;
  75. };
  76. /**
  77. * Return an array of upserted ids
  78. *
  79. * @return {object[]}
  80. */
  81. this.getUpsertedIds = function() {
  82. return bulkResult.upserted;
  83. };
  84. /**
  85. * Return the upserted id at position x
  86. *
  87. * @param {number} index the number of the upserted id to return, returns undefined if no result for passed in index
  88. * @return {object}
  89. */
  90. this.getUpsertedIdAt = function(index) {
  91. return bulkResult.upserted[index];
  92. };
  93. /**
  94. * Return raw internal result
  95. *
  96. * @return {object}
  97. */
  98. this.getRawResponse = function() {
  99. return bulkResult;
  100. };
  101. /**
  102. * Returns true if the bulk operation contains a write error
  103. *
  104. * @return {boolean}
  105. */
  106. this.hasWriteErrors = function() {
  107. return bulkResult.writeErrors.length > 0;
  108. };
  109. /**
  110. * Returns the number of write errors off the bulk operation
  111. *
  112. * @return {number}
  113. */
  114. this.getWriteErrorCount = function() {
  115. return bulkResult.writeErrors.length;
  116. };
  117. /**
  118. * Returns a specific write error object
  119. *
  120. * @param {number} index of the write error to return, returns null if there is no result for passed in index
  121. * @return {WriteError}
  122. */
  123. this.getWriteErrorAt = function(index) {
  124. if (index < bulkResult.writeErrors.length) {
  125. return bulkResult.writeErrors[index];
  126. }
  127. return null;
  128. };
  129. /**
  130. * Retrieve all write errors
  131. *
  132. * @return {object[]}
  133. */
  134. this.getWriteErrors = function() {
  135. return bulkResult.writeErrors;
  136. };
  137. /**
  138. * Retrieve lastOp if available
  139. *
  140. * @return {object}
  141. */
  142. this.getLastOp = function() {
  143. return bulkResult.lastOp;
  144. };
  145. /**
  146. * Retrieve the write concern error if any
  147. *
  148. * @return {WriteConcernError}
  149. */
  150. this.getWriteConcernError = function() {
  151. if (bulkResult.writeConcernErrors.length === 0) {
  152. return null;
  153. } else if (bulkResult.writeConcernErrors.length === 1) {
  154. // Return the error
  155. return bulkResult.writeConcernErrors[0];
  156. } else {
  157. // Combine the errors
  158. var errmsg = '';
  159. for (var i = 0; i < bulkResult.writeConcernErrors.length; i++) {
  160. var err = bulkResult.writeConcernErrors[i];
  161. errmsg = errmsg + err.errmsg;
  162. // TODO: Something better
  163. if (i === 0) errmsg = errmsg + ' and ';
  164. }
  165. return new WriteConcernError({ errmsg: errmsg, code: WRITE_CONCERN_ERROR });
  166. }
  167. };
  168. this.toJSON = function() {
  169. return bulkResult;
  170. };
  171. this.toString = function() {
  172. return 'BulkWriteResult(' + this.toJSON(bulkResult) + ')';
  173. };
  174. this.isOk = function() {
  175. return bulkResult.ok === 1;
  176. };
  177. };
  178. /**
  179. * Create a new WriteConcernError instance (INTERNAL TYPE, do not instantiate directly)
  180. *
  181. * @class
  182. * @property {number} code Write concern error code.
  183. * @property {string} errmsg Write concern error message.
  184. * @return {WriteConcernError} a WriteConcernError instance
  185. */
  186. var WriteConcernError = function(err) {
  187. if (!(this instanceof WriteConcernError)) return new WriteConcernError(err);
  188. // Define properties
  189. defineReadOnlyProperty(this, 'code', err.code);
  190. defineReadOnlyProperty(this, 'errmsg', err.errmsg);
  191. this.toJSON = function() {
  192. return { code: err.code, errmsg: err.errmsg };
  193. };
  194. this.toString = function() {
  195. return 'WriteConcernError(' + err.errmsg + ')';
  196. };
  197. };
  198. /**
  199. * Create a new WriteError instance (INTERNAL TYPE, do not instantiate directly)
  200. *
  201. * @class
  202. * @property {number} code Write concern error code.
  203. * @property {number} index Write concern error original bulk operation index.
  204. * @property {string} errmsg Write concern error message.
  205. * @return {WriteConcernError} a WriteConcernError instance
  206. */
  207. var WriteError = function(err) {
  208. if (!(this instanceof WriteError)) return new WriteError(err);
  209. // Define properties
  210. defineReadOnlyProperty(this, 'code', err.code);
  211. defineReadOnlyProperty(this, 'index', err.index);
  212. defineReadOnlyProperty(this, 'errmsg', err.errmsg);
  213. //
  214. // Define access methods
  215. this.getOperation = function() {
  216. return err.op;
  217. };
  218. this.toJSON = function() {
  219. return { code: err.code, index: err.index, errmsg: err.errmsg, op: err.op };
  220. };
  221. this.toString = function() {
  222. return 'WriteError(' + JSON.stringify(this.toJSON()) + ')';
  223. };
  224. };
  225. /**
  226. * Merges results into shared data structure
  227. * @ignore
  228. */
  229. var mergeBatchResults = function(ordered, batch, bulkResult, err, result) {
  230. // If we have an error set the result to be the err object
  231. if (err) {
  232. result = err;
  233. } else if (result && result.result) {
  234. result = result.result;
  235. } else if (result == null) {
  236. return;
  237. }
  238. // Do we have a top level error stop processing and return
  239. if (result.ok === 0 && bulkResult.ok === 1) {
  240. bulkResult.ok = 0;
  241. var writeError = {
  242. index: 0,
  243. code: result.code || 0,
  244. errmsg: result.message,
  245. op: batch.operations[0]
  246. };
  247. bulkResult.writeErrors.push(new WriteError(writeError));
  248. return;
  249. } else if (result.ok === 0 && bulkResult.ok === 0) {
  250. return;
  251. }
  252. // Deal with opTime if available
  253. if (result.opTime || result.lastOp) {
  254. var opTime = result.lastOp || result.opTime;
  255. var lastOpTS = null;
  256. var lastOpT = null;
  257. // We have a time stamp
  258. if (opTime && opTime._bsontype === 'Timestamp') {
  259. if (bulkResult.lastOp == null) {
  260. bulkResult.lastOp = opTime;
  261. } else if (opTime.greaterThan(bulkResult.lastOp)) {
  262. bulkResult.lastOp = opTime;
  263. }
  264. } else {
  265. // Existing TS
  266. if (bulkResult.lastOp) {
  267. lastOpTS =
  268. typeof bulkResult.lastOp.ts === 'number'
  269. ? Long.fromNumber(bulkResult.lastOp.ts)
  270. : bulkResult.lastOp.ts;
  271. lastOpT =
  272. typeof bulkResult.lastOp.t === 'number'
  273. ? Long.fromNumber(bulkResult.lastOp.t)
  274. : bulkResult.lastOp.t;
  275. }
  276. // Current OpTime TS
  277. var opTimeTS = typeof opTime.ts === 'number' ? Long.fromNumber(opTime.ts) : opTime.ts;
  278. var opTimeT = typeof opTime.t === 'number' ? Long.fromNumber(opTime.t) : opTime.t;
  279. // Compare the opTime's
  280. if (bulkResult.lastOp == null) {
  281. bulkResult.lastOp = opTime;
  282. } else if (opTimeTS.greaterThan(lastOpTS)) {
  283. bulkResult.lastOp = opTime;
  284. } else if (opTimeTS.equals(lastOpTS)) {
  285. if (opTimeT.greaterThan(lastOpT)) {
  286. bulkResult.lastOp = opTime;
  287. }
  288. }
  289. }
  290. }
  291. // If we have an insert Batch type
  292. if (batch.batchType === INSERT && result.n) {
  293. bulkResult.nInserted = bulkResult.nInserted + result.n;
  294. }
  295. // If we have an insert Batch type
  296. if (batch.batchType === REMOVE && result.n) {
  297. bulkResult.nRemoved = bulkResult.nRemoved + result.n;
  298. }
  299. var nUpserted = 0;
  300. // We have an array of upserted values, we need to rewrite the indexes
  301. if (Array.isArray(result.upserted)) {
  302. nUpserted = result.upserted.length;
  303. for (var i = 0; i < result.upserted.length; i++) {
  304. bulkResult.upserted.push({
  305. index: result.upserted[i].index + batch.originalZeroIndex,
  306. _id: result.upserted[i]._id
  307. });
  308. }
  309. } else if (result.upserted) {
  310. nUpserted = 1;
  311. bulkResult.upserted.push({
  312. index: batch.originalZeroIndex,
  313. _id: result.upserted
  314. });
  315. }
  316. // If we have an update Batch type
  317. if (batch.batchType === UPDATE && result.n) {
  318. var nModified = result.nModified;
  319. bulkResult.nUpserted = bulkResult.nUpserted + nUpserted;
  320. bulkResult.nMatched = bulkResult.nMatched + (result.n - nUpserted);
  321. if (typeof nModified === 'number') {
  322. bulkResult.nModified = bulkResult.nModified + nModified;
  323. } else {
  324. bulkResult.nModified = null;
  325. }
  326. }
  327. if (Array.isArray(result.writeErrors)) {
  328. for (i = 0; i < result.writeErrors.length; i++) {
  329. writeError = {
  330. index: batch.originalZeroIndex + result.writeErrors[i].index,
  331. code: result.writeErrors[i].code,
  332. errmsg: result.writeErrors[i].errmsg,
  333. op: batch.operations[result.writeErrors[i].index]
  334. };
  335. bulkResult.writeErrors.push(new WriteError(writeError));
  336. }
  337. }
  338. if (result.writeConcernError) {
  339. bulkResult.writeConcernErrors.push(new WriteConcernError(result.writeConcernError));
  340. }
  341. };
  342. //
  343. // Clone the options
  344. var cloneOptions = function(options) {
  345. var clone = {};
  346. var keys = Object.keys(options);
  347. for (var i = 0; i < keys.length; i++) {
  348. clone[keys[i]] = options[keys[i]];
  349. }
  350. return clone;
  351. };
  352. /**
  353. * Creates a new BulkWriteError
  354. *
  355. * @class
  356. * @param {Error|string|object} message The error message
  357. * @param {BulkWriteResult} result The result of the bulk write operation
  358. * @return {BulkWriteError} A BulkWriteError instance
  359. * @extends {MongoError}
  360. */
  361. const BulkWriteError = function(error, result) {
  362. var message = error.err || error.errmsg || error.errMessage || error;
  363. MongoError.call(this, message);
  364. var keys = typeof error === 'object' ? Object.keys(error) : [];
  365. for (var i = 0; i < keys.length; i++) {
  366. this[keys[i]] = error[keys[i]];
  367. }
  368. this.name = 'BulkWriteError';
  369. this.result = result;
  370. };
  371. util.inherits(BulkWriteError, MongoError);
  372. // Exports symbols
  373. exports.BulkWriteError = BulkWriteError;
  374. exports.BulkWriteResult = BulkWriteResult;
  375. exports.WriteError = WriteError;
  376. exports.Batch = Batch;
  377. exports.LegacyOp = LegacyOp;
  378. exports.mergeBatchResults = mergeBatchResults;
  379. exports.cloneOptions = cloneOptions;
  380. exports.INVALID_BSON_ERROR = INVALID_BSON_ERROR;
  381. exports.WRITE_CONCERN_ERROR = WRITE_CONCERN_ERROR;
  382. exports.MULTIPLE_ERROR = MULTIPLE_ERROR;
  383. exports.UNKNOWN_ERROR = UNKNOWN_ERROR;
  384. exports.INSERT = INSERT;
  385. exports.UPDATE = UPDATE;
  386. exports.REMOVE = REMOVE;