jsforce2 1.11.1 → 5.2.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Potentially problematic release.
This version of jsforce2 might be problematic. Click here for more details.
- package/index.js +46 -1
- package/package.json +7 -105
- package/LICENSE +0 -22
- package/README.md +0 -74
- package/bin/jsforce +0 -3
- package/bower.json +0 -30
- package/build/jsforce-api-analytics.js +0 -393
- package/build/jsforce-api-analytics.min.js +0 -2
- package/build/jsforce-api-analytics.min.js.map +0 -1
- package/build/jsforce-api-apex.js +0 -183
- package/build/jsforce-api-apex.min.js +0 -2
- package/build/jsforce-api-apex.min.js.map +0 -1
- package/build/jsforce-api-bulk.js +0 -1054
- package/build/jsforce-api-bulk.min.js +0 -2
- package/build/jsforce-api-bulk.min.js.map +0 -1
- package/build/jsforce-api-chatter.js +0 -320
- package/build/jsforce-api-chatter.min.js +0 -2
- package/build/jsforce-api-chatter.min.js.map +0 -1
- package/build/jsforce-api-metadata.js +0 -3020
- package/build/jsforce-api-metadata.min.js +0 -2
- package/build/jsforce-api-metadata.min.js.map +0 -1
- package/build/jsforce-api-soap.js +0 -403
- package/build/jsforce-api-soap.min.js +0 -2
- package/build/jsforce-api-soap.min.js.map +0 -1
- package/build/jsforce-api-streaming.js +0 -3479
- package/build/jsforce-api-streaming.min.js +0 -2
- package/build/jsforce-api-streaming.min.js.map +0 -1
- package/build/jsforce-api-tooling.js +0 -319
- package/build/jsforce-api-tooling.min.js +0 -2
- package/build/jsforce-api-tooling.min.js.map +0 -1
- package/build/jsforce-core.js +0 -25250
- package/build/jsforce-core.min.js +0 -2
- package/build/jsforce-core.min.js.map +0 -1
- package/build/jsforce.js +0 -31637
- package/build/jsforce.min.js +0 -2
- package/build/jsforce.min.js.map +0 -1
- package/core.js +0 -1
- package/lib/VERSION.js +0 -2
- package/lib/_required.js +0 -29
- package/lib/api/analytics.js +0 -387
- package/lib/api/apex.js +0 -177
- package/lib/api/bulk.js +0 -862
- package/lib/api/chatter.js +0 -314
- package/lib/api/index.js +0 -8
- package/lib/api/metadata.js +0 -848
- package/lib/api/soap.js +0 -397
- package/lib/api/streaming-extension.js +0 -136
- package/lib/api/streaming.js +0 -270
- package/lib/api/tooling.js +0 -313
- package/lib/browser/canvas.js +0 -90
- package/lib/browser/client.js +0 -241
- package/lib/browser/core.js +0 -5
- package/lib/browser/jsforce.js +0 -6
- package/lib/browser/jsonp.js +0 -52
- package/lib/browser/request.js +0 -70
- package/lib/cache.js +0 -252
- package/lib/cli/cli.js +0 -431
- package/lib/cli/repl.js +0 -337
- package/lib/connection.js +0 -1881
- package/lib/core.js +0 -16
- package/lib/csv.js +0 -50
- package/lib/date.js +0 -163
- package/lib/http-api.js +0 -300
- package/lib/jsforce.js +0 -10
- package/lib/logger.js +0 -52
- package/lib/oauth2.js +0 -206
- package/lib/process.js +0 -275
- package/lib/promise.js +0 -164
- package/lib/query.js +0 -881
- package/lib/quick-action.js +0 -90
- package/lib/record-stream.js +0 -305
- package/lib/record.js +0 -107
- package/lib/registry/file-registry.js +0 -48
- package/lib/registry/index.js +0 -3
- package/lib/registry/registry.js +0 -111
- package/lib/require.js +0 -14
- package/lib/soap.js +0 -207
- package/lib/sobject.js +0 -558
- package/lib/soql-builder.js +0 -236
- package/lib/transport.js +0 -233
|
@@ -1,1054 +0,0 @@
|
|
|
1
|
-
(function(f){if(typeof exports==="object"&&typeof module!=="undefined"){module.exports=f()}else if(typeof define==="function"&&define.amd){define([],f)}else{var g;if(typeof window!=="undefined"){g=window}else if(typeof global!=="undefined"){g=global}else if(typeof self!=="undefined"){g=self}else{g=this}g=(g.jsforce||(g.jsforce = {}));g=(g.modules||(g.modules = {}));g=(g.api||(g.api = {}));g.Bulk = f()}})(function(){var define,module,exports;return (function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c="function"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error("Cannot find module '"+i+"'");throw a.code="MODULE_NOT_FOUND",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u="function"==typeof require&&require,i=0;i<t.length;i++)o(t[i]);return o}return r})()({1:[function(require,module,exports){
|
|
2
|
-
(function (process){
|
|
3
|
-
/*global process*/
|
|
4
|
-
/**
|
|
5
|
-
* @file Manages Salesforce Bulk API related operations
|
|
6
|
-
* @author Shinichi Tomita <shinichi.tomita@gmail.com>
|
|
7
|
-
*/
|
|
8
|
-
|
|
9
|
-
'use strict';
|
|
10
|
-
|
|
11
|
-
var inherits = window.jsforce.require('inherits'),
|
|
12
|
-
stream = window.jsforce.require('readable-stream'),
|
|
13
|
-
Duplex = stream.Duplex,
|
|
14
|
-
events = window.jsforce.require('events'),
|
|
15
|
-
_ = window.jsforce.require('lodash/core'),
|
|
16
|
-
joinStreams = window.jsforce.require('multistream'),
|
|
17
|
-
jsforce = window.jsforce.require('./core'),
|
|
18
|
-
RecordStream = window.jsforce.require('./record-stream'),
|
|
19
|
-
Promise = window.jsforce.require('./promise'),
|
|
20
|
-
HttpApi = window.jsforce.require('./http-api');
|
|
21
|
-
|
|
22
|
-
/*--------------------------------------------*/
|
|
23
|
-
|
|
24
|
-
/**
|
|
25
|
-
* Class for Bulk API Job
|
|
26
|
-
*
|
|
27
|
-
* @protected
|
|
28
|
-
* @class Bulk~Job
|
|
29
|
-
* @extends events.EventEmitter
|
|
30
|
-
*
|
|
31
|
-
* @param {Bulk} bulk - Bulk API object
|
|
32
|
-
* @param {String} [type] - SObject type
|
|
33
|
-
* @param {String} [operation] - Bulk load operation ('insert', 'update', 'upsert', 'delete', or 'hardDelete')
|
|
34
|
-
* @param {Object} [options] - Options for bulk loading operation
|
|
35
|
-
* @param {String} [options.extIdField] - External ID field name (used when upsert operation).
|
|
36
|
-
* @param {String} [options.concurrencyMode] - 'Serial' or 'Parallel'. Defaults to Parallel.
|
|
37
|
-
* @param {String} [jobId] - Job ID (if already available)
|
|
38
|
-
*/
|
|
39
|
-
var Job = function(bulk, type, operation, options, jobId) {
|
|
40
|
-
this._bulk = bulk;
|
|
41
|
-
this.type = type;
|
|
42
|
-
this.operation = operation;
|
|
43
|
-
this.options = options || {};
|
|
44
|
-
this.id = jobId;
|
|
45
|
-
this.state = this.id ? 'Open' : 'Unknown';
|
|
46
|
-
this._batches = {};
|
|
47
|
-
};
|
|
48
|
-
|
|
49
|
-
inherits(Job, events.EventEmitter);
|
|
50
|
-
|
|
51
|
-
/**
|
|
52
|
-
* @typedef {Object} Bulk~JobInfo
|
|
53
|
-
* @prop {String} id - Job ID
|
|
54
|
-
* @prop {String} object - Object type name
|
|
55
|
-
* @prop {String} operation - Operation type of the job
|
|
56
|
-
* @prop {String} state - Job status
|
|
57
|
-
*/
|
|
58
|
-
|
|
59
|
-
/**
|
|
60
|
-
* Return latest jobInfo from cache
|
|
61
|
-
*
|
|
62
|
-
* @method Bulk~Job#info
|
|
63
|
-
* @param {Callback.<Bulk~JobInfo>} [callback] - Callback function
|
|
64
|
-
* @returns {Promise.<Bulk~JobInfo>}
|
|
65
|
-
*/
|
|
66
|
-
Job.prototype.info = function(callback) {
|
|
67
|
-
var self = this;
|
|
68
|
-
// if cache is not available, check the latest
|
|
69
|
-
if (!this._jobInfo) {
|
|
70
|
-
this._jobInfo = this.check();
|
|
71
|
-
}
|
|
72
|
-
return this._jobInfo.thenCall(callback);
|
|
73
|
-
};
|
|
74
|
-
|
|
75
|
-
/**
|
|
76
|
-
* Open new job and get jobinfo
|
|
77
|
-
*
|
|
78
|
-
* @method Bulk~Job#open
|
|
79
|
-
* @param {Callback.<Bulk~JobInfo>} [callback] - Callback function
|
|
80
|
-
* @returns {Promise.<Bulk~JobInfo>}
|
|
81
|
-
*/
|
|
82
|
-
Job.prototype.open = function(callback) {
|
|
83
|
-
var self = this;
|
|
84
|
-
var bulk = this._bulk;
|
|
85
|
-
var logger = bulk._logger;
|
|
86
|
-
|
|
87
|
-
// if not requested opening job
|
|
88
|
-
if (!this._jobInfo) {
|
|
89
|
-
var operation = this.operation.toLowerCase();
|
|
90
|
-
if (operation === 'harddelete') { operation = 'hardDelete'; }
|
|
91
|
-
var body = [
|
|
92
|
-
'<?xml version="1.0" encoding="UTF-8"?>',
|
|
93
|
-
'<jobInfo xmlns="http://www.force.com/2009/06/asyncapi/dataload">',
|
|
94
|
-
'<operation>' + operation + '</operation>',
|
|
95
|
-
'<object>' + this.type + '</object>',
|
|
96
|
-
(this.options.extIdField ?
|
|
97
|
-
'<externalIdFieldName>'+this.options.extIdField+'</externalIdFieldName>' :
|
|
98
|
-
''),
|
|
99
|
-
(this.options.concurrencyMode ?
|
|
100
|
-
'<concurrencyMode>'+this.options.concurrencyMode+'</concurrencyMode>' :
|
|
101
|
-
''),
|
|
102
|
-
(this.options.assignmentRuleId ?
|
|
103
|
-
'<assignmentRuleId>' + this.options.assignmentRuleId + '</assignmentRuleId>' :
|
|
104
|
-
''),
|
|
105
|
-
'<contentType>CSV</contentType>',
|
|
106
|
-
'</jobInfo>'
|
|
107
|
-
].join('');
|
|
108
|
-
|
|
109
|
-
this._jobInfo = bulk._request({
|
|
110
|
-
method : 'POST',
|
|
111
|
-
path : "/job",
|
|
112
|
-
body : body,
|
|
113
|
-
headers : {
|
|
114
|
-
"Content-Type" : "application/xml; charset=utf-8"
|
|
115
|
-
},
|
|
116
|
-
responseType: "application/xml"
|
|
117
|
-
}).then(function(res) {
|
|
118
|
-
self.emit("open", res.jobInfo);
|
|
119
|
-
self.id = res.jobInfo.id;
|
|
120
|
-
self.state = res.jobInfo.state;
|
|
121
|
-
return res.jobInfo;
|
|
122
|
-
}, function(err) {
|
|
123
|
-
self.emit("error", err);
|
|
124
|
-
throw err;
|
|
125
|
-
});
|
|
126
|
-
}
|
|
127
|
-
return this._jobInfo.thenCall(callback);
|
|
128
|
-
};
|
|
129
|
-
|
|
130
|
-
/**
|
|
131
|
-
* Create a new batch instance in the job
|
|
132
|
-
*
|
|
133
|
-
* @method Bulk~Job#createBatch
|
|
134
|
-
* @returns {Bulk~Batch}
|
|
135
|
-
*/
|
|
136
|
-
Job.prototype.createBatch = function() {
|
|
137
|
-
var batch = new Batch(this);
|
|
138
|
-
var self = this;
|
|
139
|
-
batch.on('queue', function() {
|
|
140
|
-
self._batches[batch.id] = batch;
|
|
141
|
-
});
|
|
142
|
-
return batch;
|
|
143
|
-
};
|
|
144
|
-
|
|
145
|
-
/**
|
|
146
|
-
* Get a batch instance specified by given batch ID
|
|
147
|
-
*
|
|
148
|
-
* @method Bulk~Job#batch
|
|
149
|
-
* @param {String} batchId - Batch ID
|
|
150
|
-
* @returns {Bulk~Batch}
|
|
151
|
-
*/
|
|
152
|
-
Job.prototype.batch = function(batchId) {
|
|
153
|
-
var batch = this._batches[batchId];
|
|
154
|
-
if (!batch) {
|
|
155
|
-
batch = new Batch(this, batchId);
|
|
156
|
-
this._batches[batchId] = batch;
|
|
157
|
-
}
|
|
158
|
-
return batch;
|
|
159
|
-
};
|
|
160
|
-
|
|
161
|
-
/**
|
|
162
|
-
* Check the latest job status from server
|
|
163
|
-
*
|
|
164
|
-
* @method Bulk~Job#check
|
|
165
|
-
* @param {Callback.<Bulk~JobInfo>} [callback] - Callback function
|
|
166
|
-
* @returns {Promise.<Bulk~JobInfo>}
|
|
167
|
-
*/
|
|
168
|
-
Job.prototype.check = function(callback) {
|
|
169
|
-
var self = this;
|
|
170
|
-
var bulk = this._bulk;
|
|
171
|
-
var logger = bulk._logger;
|
|
172
|
-
|
|
173
|
-
this._jobInfo = this._waitAssign().then(function() {
|
|
174
|
-
return bulk._request({
|
|
175
|
-
method : 'GET',
|
|
176
|
-
path : "/job/" + self.id,
|
|
177
|
-
responseType: "application/xml"
|
|
178
|
-
});
|
|
179
|
-
}).then(function(res) {
|
|
180
|
-
logger.debug(res.jobInfo);
|
|
181
|
-
self.id = res.jobInfo.id;
|
|
182
|
-
self.type = res.jobInfo.object;
|
|
183
|
-
self.operation = res.jobInfo.operation;
|
|
184
|
-
self.state = res.jobInfo.state;
|
|
185
|
-
return res.jobInfo;
|
|
186
|
-
});
|
|
187
|
-
return this._jobInfo.thenCall(callback);
|
|
188
|
-
};
|
|
189
|
-
|
|
190
|
-
/**
|
|
191
|
-
* Wait till the job is assigned to server
|
|
192
|
-
*
|
|
193
|
-
* @method Bulk~Job#info
|
|
194
|
-
* @param {Callback.<Bulk~JobInfo>} [callback] - Callback function
|
|
195
|
-
* @returns {Promise.<Bulk~JobInfo>}
|
|
196
|
-
*/
|
|
197
|
-
Job.prototype._waitAssign = function(callback) {
|
|
198
|
-
return (this.id ? Promise.resolve({ id: this.id }) : this.open()).thenCall(callback);
|
|
199
|
-
};
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
/**
|
|
203
|
-
* List all registered batch info in job
|
|
204
|
-
*
|
|
205
|
-
* @method Bulk~Job#list
|
|
206
|
-
* @param {Callback.<Array.<Bulk~BatchInfo>>} [callback] - Callback function
|
|
207
|
-
* @returns {Promise.<Array.<Bulk~BatchInfo>>}
|
|
208
|
-
*/
|
|
209
|
-
Job.prototype.list = function(callback) {
|
|
210
|
-
var self = this;
|
|
211
|
-
var bulk = this._bulk;
|
|
212
|
-
var logger = bulk._logger;
|
|
213
|
-
|
|
214
|
-
return this._waitAssign().then(function() {
|
|
215
|
-
return bulk._request({
|
|
216
|
-
method : 'GET',
|
|
217
|
-
path : "/job/" + self.id + "/batch",
|
|
218
|
-
responseType: "application/xml"
|
|
219
|
-
});
|
|
220
|
-
}).then(function(res) {
|
|
221
|
-
logger.debug(res.batchInfoList.batchInfo);
|
|
222
|
-
var batchInfoList = res.batchInfoList;
|
|
223
|
-
batchInfoList = _.isArray(batchInfoList.batchInfo) ? batchInfoList.batchInfo : [ batchInfoList.batchInfo ];
|
|
224
|
-
return batchInfoList;
|
|
225
|
-
}).thenCall(callback);
|
|
226
|
-
|
|
227
|
-
};
|
|
228
|
-
|
|
229
|
-
/**
|
|
230
|
-
* Close opened job
|
|
231
|
-
*
|
|
232
|
-
* @method Bulk~Job#close
|
|
233
|
-
* @param {Callback.<Bulk~JobInfo>} [callback] - Callback function
|
|
234
|
-
* @returns {Promise.<Bulk~JobInfo>}
|
|
235
|
-
*/
|
|
236
|
-
Job.prototype.close = function() {
|
|
237
|
-
var self = this;
|
|
238
|
-
return this._changeState("Closed").then(function(jobInfo) {
|
|
239
|
-
self.id = null;
|
|
240
|
-
self.emit("close", jobInfo);
|
|
241
|
-
return jobInfo;
|
|
242
|
-
}, function(err) {
|
|
243
|
-
self.emit("error", err);
|
|
244
|
-
throw err;
|
|
245
|
-
});
|
|
246
|
-
};
|
|
247
|
-
|
|
248
|
-
/**
|
|
249
|
-
* Set the status to abort
|
|
250
|
-
*
|
|
251
|
-
* @method Bulk~Job#abort
|
|
252
|
-
* @param {Callback.<Bulk~JobInfo>} [callback] - Callback function
|
|
253
|
-
* @returns {Promise.<Bulk~JobInfo>}
|
|
254
|
-
*/
|
|
255
|
-
Job.prototype.abort = function() {
|
|
256
|
-
var self = this;
|
|
257
|
-
return this._changeState("Aborted").then(function(jobInfo) {
|
|
258
|
-
self.id = null;
|
|
259
|
-
self.emit("abort", jobInfo);
|
|
260
|
-
return jobInfo;
|
|
261
|
-
}, function(err) {
|
|
262
|
-
self.emit("error", err);
|
|
263
|
-
throw err;
|
|
264
|
-
});
|
|
265
|
-
};
|
|
266
|
-
|
|
267
|
-
/**
|
|
268
|
-
* @private
|
|
269
|
-
*/
|
|
270
|
-
Job.prototype._changeState = function(state, callback) {
|
|
271
|
-
var self = this;
|
|
272
|
-
var bulk = this._bulk;
|
|
273
|
-
var logger = bulk._logger;
|
|
274
|
-
|
|
275
|
-
this._jobInfo = this._waitAssign().then(function() {
|
|
276
|
-
var body = [
|
|
277
|
-
'<?xml version="1.0" encoding="UTF-8"?>',
|
|
278
|
-
'<jobInfo xmlns="http://www.force.com/2009/06/asyncapi/dataload">',
|
|
279
|
-
'<state>' + state + '</state>',
|
|
280
|
-
'</jobInfo>'
|
|
281
|
-
].join('');
|
|
282
|
-
return bulk._request({
|
|
283
|
-
method : 'POST',
|
|
284
|
-
path : "/job/" + self.id,
|
|
285
|
-
body : body,
|
|
286
|
-
headers : {
|
|
287
|
-
"Content-Type" : "application/xml; charset=utf-8"
|
|
288
|
-
},
|
|
289
|
-
responseType: "application/xml"
|
|
290
|
-
});
|
|
291
|
-
}).then(function(res) {
|
|
292
|
-
logger.debug(res.jobInfo);
|
|
293
|
-
self.state = res.jobInfo.state;
|
|
294
|
-
return res.jobInfo;
|
|
295
|
-
});
|
|
296
|
-
return this._jobInfo.thenCall(callback);
|
|
297
|
-
|
|
298
|
-
};
|
|
299
|
-
|
|
300
|
-
|
|
301
|
-
/*--------------------------------------------*/
|
|
302
|
-
|
|
303
|
-
/**
|
|
304
|
-
* Batch (extends RecordStream)
|
|
305
|
-
*
|
|
306
|
-
* @protected
|
|
307
|
-
* @class Bulk~Batch
|
|
308
|
-
* @extends {stream.Writable}
|
|
309
|
-
* @implements {Promise.<Array.<RecordResult>>}
|
|
310
|
-
* @param {Bulk~Job} job - Bulk job object
|
|
311
|
-
* @param {String} [batchId] - Batch ID (if already available)
|
|
312
|
-
*/
|
|
313
|
-
var Batch = function(job, batchId) {
|
|
314
|
-
Batch.super_.call(this, { objectMode: true });
|
|
315
|
-
this.job = job;
|
|
316
|
-
this.id = batchId;
|
|
317
|
-
this._bulk = job._bulk;
|
|
318
|
-
this._deferred = Promise.defer();
|
|
319
|
-
this._setupDataStreams();
|
|
320
|
-
};
|
|
321
|
-
|
|
322
|
-
inherits(Batch, stream.Writable);
|
|
323
|
-
|
|
324
|
-
|
|
325
|
-
/**
|
|
326
|
-
* @private
|
|
327
|
-
*/
|
|
328
|
-
Batch.prototype._setupDataStreams = function() {
|
|
329
|
-
var batch = this;
|
|
330
|
-
var converterOptions = { nullValue : '#N/A' };
|
|
331
|
-
this._uploadStream = new RecordStream.Serializable();
|
|
332
|
-
this._uploadDataStream = this._uploadStream.stream('csv', converterOptions);
|
|
333
|
-
this._downloadStream = new RecordStream.Parsable();
|
|
334
|
-
this._downloadDataStream = this._downloadStream.stream('csv', converterOptions);
|
|
335
|
-
|
|
336
|
-
this.on('finish', function() {
|
|
337
|
-
batch._uploadStream.end();
|
|
338
|
-
});
|
|
339
|
-
this._uploadDataStream.once('readable', function() {
|
|
340
|
-
batch.job.open().then(function() {
|
|
341
|
-
// pipe upload data to batch API request stream
|
|
342
|
-
batch._uploadDataStream.pipe(batch._createRequestStream());
|
|
343
|
-
});
|
|
344
|
-
});
|
|
345
|
-
|
|
346
|
-
// duplex data stream, opened access to API programmers by Batch#stream()
|
|
347
|
-
var dataStream = this._dataStream = new Duplex();
|
|
348
|
-
dataStream._write = function(data, enc, cb) {
|
|
349
|
-
batch._uploadDataStream.write(data, enc, cb);
|
|
350
|
-
};
|
|
351
|
-
dataStream.on('finish', function() {
|
|
352
|
-
batch._uploadDataStream.end();
|
|
353
|
-
});
|
|
354
|
-
|
|
355
|
-
this._downloadDataStream.on('readable', function() {
|
|
356
|
-
dataStream.read(0);
|
|
357
|
-
});
|
|
358
|
-
this._downloadDataStream.on('end', function() {
|
|
359
|
-
dataStream.push(null);
|
|
360
|
-
});
|
|
361
|
-
dataStream._read = function(size) {
|
|
362
|
-
var chunk;
|
|
363
|
-
while ((chunk = batch._downloadDataStream.read()) !== null) {
|
|
364
|
-
dataStream.push(chunk);
|
|
365
|
-
}
|
|
366
|
-
};
|
|
367
|
-
};
|
|
368
|
-
|
|
369
|
-
/**
|
|
370
|
-
* Connect batch API and create stream instance of request/response
|
|
371
|
-
*
|
|
372
|
-
* @private
|
|
373
|
-
* @returns {stream.Duplex}
|
|
374
|
-
*/
|
|
375
|
-
Batch.prototype._createRequestStream = function() {
|
|
376
|
-
var batch = this;
|
|
377
|
-
var bulk = batch._bulk;
|
|
378
|
-
var logger = bulk._logger;
|
|
379
|
-
|
|
380
|
-
return bulk._request({
|
|
381
|
-
method : 'POST',
|
|
382
|
-
path : "/job/" + batch.job.id + "/batch",
|
|
383
|
-
headers: {
|
|
384
|
-
"Content-Type": "text/csv"
|
|
385
|
-
},
|
|
386
|
-
responseType: "application/xml"
|
|
387
|
-
}, function(err, res) {
|
|
388
|
-
if (err) {
|
|
389
|
-
batch.emit('error', err);
|
|
390
|
-
} else {
|
|
391
|
-
logger.debug(res.batchInfo);
|
|
392
|
-
batch.id = res.batchInfo.id;
|
|
393
|
-
batch.emit('queue', res.batchInfo);
|
|
394
|
-
}
|
|
395
|
-
}).stream();
|
|
396
|
-
};
|
|
397
|
-
|
|
398
|
-
/**
|
|
399
|
-
* Implementation of Writable
|
|
400
|
-
*
|
|
401
|
-
* @override
|
|
402
|
-
* @private
|
|
403
|
-
*/
|
|
404
|
-
Batch.prototype._write = function(record, enc, cb) {
|
|
405
|
-
record = _.clone(record);
|
|
406
|
-
if (this.job.operation === "insert") {
|
|
407
|
-
delete record.Id;
|
|
408
|
-
} else if (this.job.operation === "delete") {
|
|
409
|
-
record = { Id: record.Id };
|
|
410
|
-
}
|
|
411
|
-
delete record.type;
|
|
412
|
-
delete record.attributes;
|
|
413
|
-
this._uploadStream.write(record, enc, cb);
|
|
414
|
-
};
|
|
415
|
-
|
|
416
|
-
/**
|
|
417
|
-
* Returns duplex stream which accepts CSV data input and batch result output
|
|
418
|
-
*
|
|
419
|
-
* @returns {stream.Duplex}
|
|
420
|
-
*/
|
|
421
|
-
Batch.prototype.stream = function() {
|
|
422
|
-
return this._dataStream;
|
|
423
|
-
};
|
|
424
|
-
|
|
425
|
-
/**
|
|
426
|
-
* Execute batch operation
|
|
427
|
-
*
|
|
428
|
-
* @method Bulk~Batch#execute
|
|
429
|
-
* @param {Array.<Record>|stream.Stream|String} [input] - Input source for batch operation. Accepts array of records, CSV string, and CSV data input stream in insert/update/upsert/delete/hardDelete operation, SOQL string in query operation.
|
|
430
|
-
* @param {Callback.<Array.<RecordResult>|Array.<BatchResultInfo>>} [callback] - Callback function
|
|
431
|
-
* @returns {Bulk~Batch}
|
|
432
|
-
*/
|
|
433
|
-
Batch.prototype.run =
|
|
434
|
-
Batch.prototype.exec =
|
|
435
|
-
Batch.prototype.execute = function(input, callback) {
|
|
436
|
-
var self = this;
|
|
437
|
-
|
|
438
|
-
if (typeof input === 'function') { // if input argument is omitted
|
|
439
|
-
callback = input;
|
|
440
|
-
input = null;
|
|
441
|
-
}
|
|
442
|
-
|
|
443
|
-
// if batch is already executed
|
|
444
|
-
if (this._result) {
|
|
445
|
-
throw new Error("Batch already executed.");
|
|
446
|
-
}
|
|
447
|
-
|
|
448
|
-
var rdeferred = Promise.defer();
|
|
449
|
-
this._result = rdeferred.promise;
|
|
450
|
-
this._result.then(function(res) {
|
|
451
|
-
self._deferred.resolve(res);
|
|
452
|
-
}, function(err) {
|
|
453
|
-
self._deferred.reject(err);
|
|
454
|
-
});
|
|
455
|
-
this.once('response', function(res) {
|
|
456
|
-
rdeferred.resolve(res);
|
|
457
|
-
});
|
|
458
|
-
this.once('error', function(err) {
|
|
459
|
-
rdeferred.reject(err);
|
|
460
|
-
});
|
|
461
|
-
|
|
462
|
-
if (_.isObject(input) && _.isFunction(input.pipe)) { // if input has stream.Readable interface
|
|
463
|
-
input.pipe(this._dataStream);
|
|
464
|
-
} else {
|
|
465
|
-
var data;
|
|
466
|
-
if (_.isArray(input)) {
|
|
467
|
-
_.forEach(input, function(record) {
|
|
468
|
-
Object.keys(record).forEach(function(key) {
|
|
469
|
-
if (typeof record[key] === 'boolean') {
|
|
470
|
-
record[key] = String(record[key])
|
|
471
|
-
}
|
|
472
|
-
})
|
|
473
|
-
self.write(record);
|
|
474
|
-
});
|
|
475
|
-
self.end();
|
|
476
|
-
} else if (_.isString(input)){
|
|
477
|
-
data = input;
|
|
478
|
-
this._dataStream.write(data, 'utf8');
|
|
479
|
-
this._dataStream.end();
|
|
480
|
-
}
|
|
481
|
-
}
|
|
482
|
-
|
|
483
|
-
// return Batch instance for chaining
|
|
484
|
-
return this.thenCall(callback);
|
|
485
|
-
};
|
|
486
|
-
|
|
487
|
-
/**
|
|
488
|
-
* Promise/A+ interface
|
|
489
|
-
* http://promises-aplus.github.io/promises-spec/
|
|
490
|
-
*
|
|
491
|
-
* Delegate to deferred promise, return promise instance for batch result
|
|
492
|
-
*
|
|
493
|
-
* @method Bulk~Batch#then
|
|
494
|
-
*/
|
|
495
|
-
Batch.prototype.then = function(onResolved, onReject, onProgress) {
|
|
496
|
-
return this._deferred.promise.then(onResolved, onReject, onProgress);
|
|
497
|
-
};
|
|
498
|
-
|
|
499
|
-
/**
|
|
500
|
-
* Promise/A+ extension
|
|
501
|
-
* Call "then" using given node-style callback function
|
|
502
|
-
*
|
|
503
|
-
* @method Bulk~Batch#thenCall
|
|
504
|
-
*/
|
|
505
|
-
Batch.prototype.thenCall = function(callback) {
|
|
506
|
-
if (_.isFunction(callback)) {
|
|
507
|
-
this.then(function(res) {
|
|
508
|
-
process.nextTick(function() {
|
|
509
|
-
callback(null, res);
|
|
510
|
-
});
|
|
511
|
-
}, function(err) {
|
|
512
|
-
process.nextTick(function() {
|
|
513
|
-
callback(err);
|
|
514
|
-
});
|
|
515
|
-
});
|
|
516
|
-
}
|
|
517
|
-
return this;
|
|
518
|
-
};
|
|
519
|
-
|
|
520
|
-
/**
|
|
521
|
-
* @typedef {Object} Bulk~BatchInfo
|
|
522
|
-
* @prop {String} id - Batch ID
|
|
523
|
-
* @prop {String} jobId - Job ID
|
|
524
|
-
* @prop {String} state - Batch state
|
|
525
|
-
* @prop {String} stateMessage - Batch state message
|
|
526
|
-
*/
|
|
527
|
-
|
|
528
|
-
/**
|
|
529
|
-
* Check the latest batch status in server
|
|
530
|
-
*
|
|
531
|
-
* @method Bulk~Batch#check
|
|
532
|
-
* @param {Callback.<Bulk~BatchInfo>} [callback] - Callback function
|
|
533
|
-
* @returns {Promise.<Bulk~BatchInfo>}
|
|
534
|
-
*/
|
|
535
|
-
Batch.prototype.check = function(callback) {
|
|
536
|
-
var self = this;
|
|
537
|
-
var bulk = this._bulk;
|
|
538
|
-
var logger = bulk._logger;
|
|
539
|
-
var jobId = this.job.id;
|
|
540
|
-
var batchId = this.id;
|
|
541
|
-
|
|
542
|
-
if (!jobId || !batchId) {
|
|
543
|
-
throw new Error("Batch not started.");
|
|
544
|
-
}
|
|
545
|
-
return bulk._request({
|
|
546
|
-
method : 'GET',
|
|
547
|
-
path : "/job/" + jobId + "/batch/" + batchId,
|
|
548
|
-
responseType: "application/xml"
|
|
549
|
-
}).then(function(res) {
|
|
550
|
-
logger.debug(res.batchInfo);
|
|
551
|
-
return res.batchInfo;
|
|
552
|
-
}).thenCall(callback);
|
|
553
|
-
};
|
|
554
|
-
|
|
555
|
-
|
|
556
|
-
/**
|
|
557
|
-
* Polling the batch result and retrieve
|
|
558
|
-
*
|
|
559
|
-
* @method Bulk~Batch#poll
|
|
560
|
-
* @param {Number} interval - Polling interval in milliseconds
|
|
561
|
-
* @param {Number} timeout - Polling timeout in milliseconds
|
|
562
|
-
*/
|
|
563
|
-
Batch.prototype.poll = function(interval, timeout) {
|
|
564
|
-
var self = this;
|
|
565
|
-
var jobId = this.job.id;
|
|
566
|
-
var batchId = this.id;
|
|
567
|
-
|
|
568
|
-
if (!jobId || !batchId) {
|
|
569
|
-
throw new Error("Batch not started.");
|
|
570
|
-
}
|
|
571
|
-
var startTime = new Date().getTime();
|
|
572
|
-
var poll = function() {
|
|
573
|
-
var now = new Date().getTime();
|
|
574
|
-
if (startTime + timeout < now) {
|
|
575
|
-
var err = new Error("Polling time out. Job Id = " + jobId + " , batch Id = " + batchId);
|
|
576
|
-
err.name = 'PollingTimeout';
|
|
577
|
-
err.jobId = jobId;
|
|
578
|
-
err.batchId = batchId;
|
|
579
|
-
self.emit('error', err);
|
|
580
|
-
return;
|
|
581
|
-
}
|
|
582
|
-
self.check(function(err, res) {
|
|
583
|
-
if (err) {
|
|
584
|
-
self.emit('error', err);
|
|
585
|
-
} else {
|
|
586
|
-
if (res.state === "Failed") {
|
|
587
|
-
if (parseInt(res.numberRecordsProcessed, 10) > 0) {
|
|
588
|
-
self.retrieve();
|
|
589
|
-
} else {
|
|
590
|
-
self.emit('error', new Error(res.stateMessage));
|
|
591
|
-
}
|
|
592
|
-
} else if (res.state === "Completed") {
|
|
593
|
-
self.retrieve();
|
|
594
|
-
} else {
|
|
595
|
-
self.emit('progress', res);
|
|
596
|
-
setTimeout(poll, interval);
|
|
597
|
-
}
|
|
598
|
-
}
|
|
599
|
-
});
|
|
600
|
-
};
|
|
601
|
-
setTimeout(poll, interval);
|
|
602
|
-
};
|
|
603
|
-
|
|
604
|
-
/**
|
|
605
|
-
* @typedef {Object} Bulk~BatchResultInfo
|
|
606
|
-
* @prop {String} id - Batch result ID
|
|
607
|
-
* @prop {String} batchId - Batch ID which includes this batch result.
|
|
608
|
-
* @prop {String} jobId - Job ID which includes this batch result.
|
|
609
|
-
*/
|
|
610
|
-
|
|
611
|
-
/**
|
|
612
|
-
* Retrieve batch result
|
|
613
|
-
*
|
|
614
|
-
* @method Bulk~Batch#retrieve
|
|
615
|
-
* @param {Callback.<Array.<RecordResult>|Array.<Bulk~BatchResultInfo>>} [callback] - Callback function
|
|
616
|
-
* @returns {Promise.<Array.<RecordResult>|Array.<Bulk~BatchResultInfo>>}
|
|
617
|
-
*/
|
|
618
|
-
Batch.prototype.retrieve = function(callback) {
|
|
619
|
-
var self = this;
|
|
620
|
-
var bulk = this._bulk;
|
|
621
|
-
var jobId = this.job.id;
|
|
622
|
-
var job = this.job;
|
|
623
|
-
var batchId = this.id;
|
|
624
|
-
|
|
625
|
-
if (!jobId || !batchId) {
|
|
626
|
-
throw new Error("Batch not started.");
|
|
627
|
-
}
|
|
628
|
-
|
|
629
|
-
return job.info().then(function(jobInfo) {
|
|
630
|
-
return bulk._request({
|
|
631
|
-
method : 'GET',
|
|
632
|
-
path : "/job/" + jobId + "/batch/" + batchId + "/result"
|
|
633
|
-
});
|
|
634
|
-
}).then(function(res) {
|
|
635
|
-
var results;
|
|
636
|
-
if (job.operation === 'query') {
|
|
637
|
-
var conn = bulk._conn;
|
|
638
|
-
var resultIds = res['result-list'].result;
|
|
639
|
-
results = res['result-list'].result;
|
|
640
|
-
results = _.map(_.isArray(results) ? results : [ results ], function(id) {
|
|
641
|
-
return {
|
|
642
|
-
id: id,
|
|
643
|
-
batchId: batchId,
|
|
644
|
-
jobId: jobId
|
|
645
|
-
};
|
|
646
|
-
});
|
|
647
|
-
} else {
|
|
648
|
-
results = _.map(res, function(ret) {
|
|
649
|
-
return {
|
|
650
|
-
id: ret.Id || null,
|
|
651
|
-
success: ret.Success === "true",
|
|
652
|
-
errors: ret.Error ? [ ret.Error ] : []
|
|
653
|
-
};
|
|
654
|
-
});
|
|
655
|
-
}
|
|
656
|
-
self.emit('response', results);
|
|
657
|
-
return results;
|
|
658
|
-
}).fail(function(err) {
|
|
659
|
-
self.emit('error', err);
|
|
660
|
-
throw err;
|
|
661
|
-
}).thenCall(callback);
|
|
662
|
-
};
|
|
663
|
-
|
|
664
|
-
/**
|
|
665
|
-
* Fetch query result as a record stream
|
|
666
|
-
* @param {String} resultId - Result id
|
|
667
|
-
* @returns {RecordStream} - Record stream, convertible to CSV data stream
|
|
668
|
-
*/
|
|
669
|
-
Batch.prototype.result = function(resultId) {
|
|
670
|
-
var jobId = this.job.id;
|
|
671
|
-
var batchId = this.id;
|
|
672
|
-
if (!jobId || !batchId) {
|
|
673
|
-
throw new Error("Batch not started.");
|
|
674
|
-
}
|
|
675
|
-
var resultStream = new RecordStream.Parsable();
|
|
676
|
-
var resultDataStream = resultStream.stream('csv');
|
|
677
|
-
var reqStream = this._bulk._request({
|
|
678
|
-
method : 'GET',
|
|
679
|
-
path : "/job/" + jobId + "/batch/" + batchId + "/result/" + resultId,
|
|
680
|
-
responseType: "application/octet-stream"
|
|
681
|
-
}).stream().pipe(resultDataStream);
|
|
682
|
-
return resultStream;
|
|
683
|
-
};
|
|
684
|
-
|
|
685
|
-
/*--------------------------------------------*/
|
|
686
|
-
/**
|
|
687
|
-
* @private
|
|
688
|
-
*/
|
|
689
|
-
var BulkApi = function() {
|
|
690
|
-
BulkApi.super_.apply(this, arguments);
|
|
691
|
-
};
|
|
692
|
-
|
|
693
|
-
inherits(BulkApi, HttpApi);
|
|
694
|
-
|
|
695
|
-
BulkApi.prototype.beforeSend = function(request) {
|
|
696
|
-
request.headers = request.headers || {};
|
|
697
|
-
request.headers["X-SFDC-SESSION"] = this._conn.accessToken;
|
|
698
|
-
};
|
|
699
|
-
|
|
700
|
-
BulkApi.prototype.isSessionExpired = function(response) {
|
|
701
|
-
return response.statusCode === 400 &&
|
|
702
|
-
/<exceptionCode>InvalidSessionId<\/exceptionCode>/.test(response.body);
|
|
703
|
-
};
|
|
704
|
-
|
|
705
|
-
BulkApi.prototype.hasErrorInResponseBody = function(body) {
|
|
706
|
-
return !!body.error;
|
|
707
|
-
};
|
|
708
|
-
|
|
709
|
-
BulkApi.prototype.parseError = function(body) {
|
|
710
|
-
return {
|
|
711
|
-
errorCode: body.error.exceptionCode,
|
|
712
|
-
message: body.error.exceptionMessage
|
|
713
|
-
};
|
|
714
|
-
};
|
|
715
|
-
|
|
716
|
-
/*--------------------------------------------*/
|
|
717
|
-
|
|
718
|
-
/**
|
|
719
|
-
* Class for Bulk API
|
|
720
|
-
*
|
|
721
|
-
* @class
|
|
722
|
-
* @param {Connection} conn - Connection object
|
|
723
|
-
*/
|
|
724
|
-
var Bulk = function(conn) {
|
|
725
|
-
this._conn = conn;
|
|
726
|
-
this._logger = conn._logger;
|
|
727
|
-
};
|
|
728
|
-
|
|
729
|
-
/**
|
|
730
|
-
* Polling interval in milliseconds
|
|
731
|
-
* @type {Number}
|
|
732
|
-
*/
|
|
733
|
-
Bulk.prototype.pollInterval = 1000;
|
|
734
|
-
|
|
735
|
-
/**
|
|
736
|
-
* Polling timeout in milliseconds
|
|
737
|
-
* @type {Number}
|
|
738
|
-
*/
|
|
739
|
-
Bulk.prototype.pollTimeout = 10000;
|
|
740
|
-
|
|
741
|
-
/** @private **/
|
|
742
|
-
Bulk.prototype._request = function(request, callback) {
|
|
743
|
-
var conn = this._conn;
|
|
744
|
-
request = _.clone(request);
|
|
745
|
-
var baseUrl = [ conn.instanceUrl, "services/async", conn.version ].join('/');
|
|
746
|
-
request.url = baseUrl + request.path;
|
|
747
|
-
var options = { responseType: request.responseType };
|
|
748
|
-
delete request.path;
|
|
749
|
-
delete request.responseType;
|
|
750
|
-
return new BulkApi(this._conn, options).request(request).thenCall(callback);
|
|
751
|
-
};
|
|
752
|
-
|
|
753
|
-
/**
|
|
754
|
-
* Create and start bulkload job and batch
|
|
755
|
-
*
|
|
756
|
-
* @param {String} type - SObject type
|
|
757
|
-
* @param {String} operation - Bulk load operation ('insert', 'update', 'upsert', 'delete', or 'hardDelete')
|
|
758
|
-
* @param {Object} [options] - Options for bulk loading operation
|
|
759
|
-
* @param {String} [options.extIdField] - External ID field name (used when upsert operation).
|
|
760
|
-
* @param {String} [options.concurrencyMode] - 'Serial' or 'Parallel'. Defaults to Parallel.
|
|
761
|
-
* @param {Array.<Record>|stream.Stream|String} [input] - Input source for bulkload. Accepts array of records, CSV string, and CSV data input stream in insert/update/upsert/delete/hardDelete operation, SOQL string in query operation.
|
|
762
|
-
* @param {Callback.<Array.<RecordResult>|Array.<Bulk~BatchResultInfo>>} [callback] - Callback function
|
|
763
|
-
* @returns {Bulk~Batch}
|
|
764
|
-
*/
|
|
765
|
-
Bulk.prototype.load = function(type, operation, options, input, callback) {
|
|
766
|
-
var self = this;
|
|
767
|
-
if (!type || !operation) {
|
|
768
|
-
throw new Error("Insufficient arguments. At least, 'type' and 'operation' are required.");
|
|
769
|
-
}
|
|
770
|
-
if (!_.isObject(options) || options.constructor !== Object) { // when options is not plain hash object, it is omitted
|
|
771
|
-
callback = input;
|
|
772
|
-
input = options;
|
|
773
|
-
options = null;
|
|
774
|
-
}
|
|
775
|
-
var job = this.createJob(type, operation, options);
|
|
776
|
-
job.once('error', function (error) {
|
|
777
|
-
if (batch) {
|
|
778
|
-
batch.emit('error', error); // pass job error to batch
|
|
779
|
-
}
|
|
780
|
-
});
|
|
781
|
-
var batch = job.createBatch();
|
|
782
|
-
var cleanup = function() {
|
|
783
|
-
batch = null;
|
|
784
|
-
job.close();
|
|
785
|
-
};
|
|
786
|
-
var cleanupOnError = function(err) {
|
|
787
|
-
if (err.name !== 'PollingTimeout') {
|
|
788
|
-
cleanup();
|
|
789
|
-
}
|
|
790
|
-
};
|
|
791
|
-
batch.on('response', cleanup);
|
|
792
|
-
batch.on('error', cleanupOnError);
|
|
793
|
-
batch.on('queue', function() { batch.poll(self.pollInterval, self.pollTimeout); });
|
|
794
|
-
return batch.execute(input, callback);
|
|
795
|
-
};
|
|
796
|
-
|
|
797
|
-
/**
|
|
798
|
-
* Execute bulk query and get record stream
|
|
799
|
-
*
|
|
800
|
-
* @param {String} soql - SOQL to execute in bulk job
|
|
801
|
-
* @returns {RecordStream.Parsable} - Record stream, convertible to CSV data stream
|
|
802
|
-
*/
|
|
803
|
-
Bulk.prototype.query = function(soql) {
|
|
804
|
-
var m = soql.replace(/\([\s\S]+\)/g, '').match(/FROM\s+(\w+)/i);
|
|
805
|
-
if (!m) {
|
|
806
|
-
throw new Error("No sobject type found in query, maybe caused by invalid SOQL.");
|
|
807
|
-
}
|
|
808
|
-
var type = m[1];
|
|
809
|
-
var self = this;
|
|
810
|
-
var recordStream = new RecordStream.Parsable();
|
|
811
|
-
var dataStream = recordStream.stream('csv');
|
|
812
|
-
this.load(type, "query", soql).then(function(results) {
|
|
813
|
-
var streams = results.map(function(result) {
|
|
814
|
-
return self
|
|
815
|
-
.job(result.jobId)
|
|
816
|
-
.batch(result.batchId)
|
|
817
|
-
.result(result.id)
|
|
818
|
-
.stream();
|
|
819
|
-
});
|
|
820
|
-
|
|
821
|
-
joinStreams(streams).pipe(dataStream);
|
|
822
|
-
}).fail(function(err) {
|
|
823
|
-
recordStream.emit('error', err);
|
|
824
|
-
});
|
|
825
|
-
return recordStream;
|
|
826
|
-
};
|
|
827
|
-
|
|
828
|
-
|
|
829
|
-
/**
|
|
830
|
-
* Create a new job instance
|
|
831
|
-
*
|
|
832
|
-
* @param {String} type - SObject type
|
|
833
|
-
* @param {String} operation - Bulk load operation ('insert', 'update', 'upsert', 'delete', 'hardDelete', or 'query')
|
|
834
|
-
* @param {Object} [options] - Options for bulk loading operation
|
|
835
|
-
* @returns {Bulk~Job}
|
|
836
|
-
*/
|
|
837
|
-
Bulk.prototype.createJob = function(type, operation, options) {
|
|
838
|
-
return new Job(this, type, operation, options);
|
|
839
|
-
};
|
|
840
|
-
|
|
841
|
-
/**
|
|
842
|
-
* Get a job instance specified by given job ID
|
|
843
|
-
*
|
|
844
|
-
* @param {String} jobId - Job ID
|
|
845
|
-
* @returns {Bulk~Job}
|
|
846
|
-
*/
|
|
847
|
-
Bulk.prototype.job = function(jobId) {
|
|
848
|
-
return new Job(this, null, null, null, jobId);
|
|
849
|
-
};
|
|
850
|
-
|
|
851
|
-
|
|
852
|
-
/*--------------------------------------------*/
|
|
853
|
-
/*
|
|
854
|
-
* Register hook in connection instantiation for dynamically adding this API module features
|
|
855
|
-
*/
|
|
856
|
-
jsforce.on('connection:new', function(conn) {
|
|
857
|
-
conn.bulk = new Bulk(conn);
|
|
858
|
-
});
|
|
859
|
-
|
|
860
|
-
|
|
861
|
-
module.exports = Bulk;
|
|
862
|
-
|
|
863
|
-
}).call(this,require('_process'))
|
|
864
|
-
|
|
865
|
-
},{"_process":2}],2:[function(require,module,exports){
|
|
866
|
-
// shim for using process in browser
|
|
867
|
-
var process = module.exports = {};
|
|
868
|
-
|
|
869
|
-
// cached from whatever global is present so that test runners that stub it
|
|
870
|
-
// don't break things. But we need to wrap it in a try catch in case it is
|
|
871
|
-
// wrapped in strict mode code which doesn't define any globals. It's inside a
|
|
872
|
-
// function because try/catches deoptimize in certain engines.
|
|
873
|
-
|
|
874
|
-
var cachedSetTimeout;
|
|
875
|
-
var cachedClearTimeout;
|
|
876
|
-
|
|
877
|
-
function defaultSetTimout() {
|
|
878
|
-
throw new Error('setTimeout has not been defined');
|
|
879
|
-
}
|
|
880
|
-
function defaultClearTimeout () {
|
|
881
|
-
throw new Error('clearTimeout has not been defined');
|
|
882
|
-
}
|
|
883
|
-
(function () {
|
|
884
|
-
try {
|
|
885
|
-
if (typeof setTimeout === 'function') {
|
|
886
|
-
cachedSetTimeout = setTimeout;
|
|
887
|
-
} else {
|
|
888
|
-
cachedSetTimeout = defaultSetTimout;
|
|
889
|
-
}
|
|
890
|
-
} catch (e) {
|
|
891
|
-
cachedSetTimeout = defaultSetTimout;
|
|
892
|
-
}
|
|
893
|
-
try {
|
|
894
|
-
if (typeof clearTimeout === 'function') {
|
|
895
|
-
cachedClearTimeout = clearTimeout;
|
|
896
|
-
} else {
|
|
897
|
-
cachedClearTimeout = defaultClearTimeout;
|
|
898
|
-
}
|
|
899
|
-
} catch (e) {
|
|
900
|
-
cachedClearTimeout = defaultClearTimeout;
|
|
901
|
-
}
|
|
902
|
-
} ())
|
|
903
|
-
function runTimeout(fun) {
|
|
904
|
-
if (cachedSetTimeout === setTimeout) {
|
|
905
|
-
//normal enviroments in sane situations
|
|
906
|
-
return setTimeout(fun, 0);
|
|
907
|
-
}
|
|
908
|
-
// if setTimeout wasn't available but was latter defined
|
|
909
|
-
if ((cachedSetTimeout === defaultSetTimout || !cachedSetTimeout) && setTimeout) {
|
|
910
|
-
cachedSetTimeout = setTimeout;
|
|
911
|
-
return setTimeout(fun, 0);
|
|
912
|
-
}
|
|
913
|
-
try {
|
|
914
|
-
// when when somebody has screwed with setTimeout but no I.E. maddness
|
|
915
|
-
return cachedSetTimeout(fun, 0);
|
|
916
|
-
} catch(e){
|
|
917
|
-
try {
|
|
918
|
-
// When we are in I.E. but the script has been evaled so I.E. doesn't trust the global object when called normally
|
|
919
|
-
return cachedSetTimeout.call(null, fun, 0);
|
|
920
|
-
} catch(e){
|
|
921
|
-
// same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error
|
|
922
|
-
return cachedSetTimeout.call(this, fun, 0);
|
|
923
|
-
}
|
|
924
|
-
}
|
|
925
|
-
|
|
926
|
-
|
|
927
|
-
}
|
|
928
|
-
function runClearTimeout(marker) {
|
|
929
|
-
if (cachedClearTimeout === clearTimeout) {
|
|
930
|
-
//normal enviroments in sane situations
|
|
931
|
-
return clearTimeout(marker);
|
|
932
|
-
}
|
|
933
|
-
// if clearTimeout wasn't available but was latter defined
|
|
934
|
-
if ((cachedClearTimeout === defaultClearTimeout || !cachedClearTimeout) && clearTimeout) {
|
|
935
|
-
cachedClearTimeout = clearTimeout;
|
|
936
|
-
return clearTimeout(marker);
|
|
937
|
-
}
|
|
938
|
-
try {
|
|
939
|
-
// when when somebody has screwed with setTimeout but no I.E. maddness
|
|
940
|
-
return cachedClearTimeout(marker);
|
|
941
|
-
} catch (e){
|
|
942
|
-
try {
|
|
943
|
-
// When we are in I.E. but the script has been evaled so I.E. doesn't trust the global object when called normally
|
|
944
|
-
return cachedClearTimeout.call(null, marker);
|
|
945
|
-
} catch (e){
|
|
946
|
-
// same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error.
|
|
947
|
-
// Some versions of I.E. have different rules for clearTimeout vs setTimeout
|
|
948
|
-
return cachedClearTimeout.call(this, marker);
|
|
949
|
-
}
|
|
950
|
-
}
|
|
951
|
-
|
|
952
|
-
|
|
953
|
-
|
|
954
|
-
}
|
|
955
|
-
var queue = [];
|
|
956
|
-
var draining = false;
|
|
957
|
-
var currentQueue;
|
|
958
|
-
var queueIndex = -1;
|
|
959
|
-
|
|
960
|
-
function cleanUpNextTick() {
|
|
961
|
-
if (!draining || !currentQueue) {
|
|
962
|
-
return;
|
|
963
|
-
}
|
|
964
|
-
draining = false;
|
|
965
|
-
if (currentQueue.length) {
|
|
966
|
-
queue = currentQueue.concat(queue);
|
|
967
|
-
} else {
|
|
968
|
-
queueIndex = -1;
|
|
969
|
-
}
|
|
970
|
-
if (queue.length) {
|
|
971
|
-
drainQueue();
|
|
972
|
-
}
|
|
973
|
-
}
|
|
974
|
-
|
|
975
|
-
function drainQueue() {
|
|
976
|
-
if (draining) {
|
|
977
|
-
return;
|
|
978
|
-
}
|
|
979
|
-
var timeout = runTimeout(cleanUpNextTick);
|
|
980
|
-
draining = true;
|
|
981
|
-
|
|
982
|
-
var len = queue.length;
|
|
983
|
-
while(len) {
|
|
984
|
-
currentQueue = queue;
|
|
985
|
-
queue = [];
|
|
986
|
-
while (++queueIndex < len) {
|
|
987
|
-
if (currentQueue) {
|
|
988
|
-
currentQueue[queueIndex].run();
|
|
989
|
-
}
|
|
990
|
-
}
|
|
991
|
-
queueIndex = -1;
|
|
992
|
-
len = queue.length;
|
|
993
|
-
}
|
|
994
|
-
currentQueue = null;
|
|
995
|
-
draining = false;
|
|
996
|
-
runClearTimeout(timeout);
|
|
997
|
-
}
|
|
998
|
-
|
|
999
|
-
process.nextTick = function (fun) {
|
|
1000
|
-
var args = new Array(arguments.length - 1);
|
|
1001
|
-
if (arguments.length > 1) {
|
|
1002
|
-
for (var i = 1; i < arguments.length; i++) {
|
|
1003
|
-
args[i - 1] = arguments[i];
|
|
1004
|
-
}
|
|
1005
|
-
}
|
|
1006
|
-
queue.push(new Item(fun, args));
|
|
1007
|
-
if (queue.length === 1 && !draining) {
|
|
1008
|
-
runTimeout(drainQueue);
|
|
1009
|
-
}
|
|
1010
|
-
};
|
|
1011
|
-
|
|
1012
|
-
// v8 likes predictible objects
|
|
1013
|
-
function Item(fun, array) {
|
|
1014
|
-
this.fun = fun;
|
|
1015
|
-
this.array = array;
|
|
1016
|
-
}
|
|
1017
|
-
Item.prototype.run = function () {
|
|
1018
|
-
this.fun.apply(null, this.array);
|
|
1019
|
-
};
|
|
1020
|
-
process.title = 'browser';
|
|
1021
|
-
process.browser = true;
|
|
1022
|
-
process.env = {};
|
|
1023
|
-
process.argv = [];
|
|
1024
|
-
process.version = ''; // empty string to avoid regexp issues
|
|
1025
|
-
process.versions = {};
|
|
1026
|
-
|
|
1027
|
-
function noop() {}
|
|
1028
|
-
|
|
1029
|
-
process.on = noop;
|
|
1030
|
-
process.addListener = noop;
|
|
1031
|
-
process.once = noop;
|
|
1032
|
-
process.off = noop;
|
|
1033
|
-
process.removeListener = noop;
|
|
1034
|
-
process.removeAllListeners = noop;
|
|
1035
|
-
process.emit = noop;
|
|
1036
|
-
process.prependListener = noop;
|
|
1037
|
-
process.prependOnceListener = noop;
|
|
1038
|
-
|
|
1039
|
-
process.listeners = function (name) { return [] }
|
|
1040
|
-
|
|
1041
|
-
process.binding = function (name) {
|
|
1042
|
-
throw new Error('process.binding is not supported');
|
|
1043
|
-
};
|
|
1044
|
-
|
|
1045
|
-
process.cwd = function () { return '/' };
|
|
1046
|
-
process.chdir = function (dir) {
|
|
1047
|
-
throw new Error('process.chdir is not supported');
|
|
1048
|
-
};
|
|
1049
|
-
process.umask = function() { return 0; };
|
|
1050
|
-
|
|
1051
|
-
},{}]},{},[1])(1)
|
|
1052
|
-
});
|
|
1053
|
-
|
|
1054
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","lib/api/bulk.js","node_modules/process/browser.js"],"names":[],"mappings":"AAAA;;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AC31BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c=\"function\"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error(\"Cannot find module '\"+i+\"'\");throw a.code=\"MODULE_NOT_FOUND\",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u=\"function\"==typeof require&&require,i=0;i<t.length;i++)o(t[i]);return o}return r})()","/*global process*/\n/**\n * @file Manages Salesforce Bulk API related operations\n * @author Shinichi Tomita <shinichi.tomita@gmail.com>\n */\n\n'use strict';\n\nvar inherits     = window.jsforce.require('inherits'),\n    stream       = window.jsforce.require('readable-stream'),\n    Duplex       = stream.Duplex,\n    events       = window.jsforce.require('events'),\n    _            = window.jsforce.require('lodash/core'),\n    joinStreams  = window.jsforce.require('multistream'),\n    jsforce      = window.jsforce.require('./core'),\n    RecordStream = window.jsforce.require('./record-stream'),\n    Promise      = window.jsforce.require('./promise'),\n    HttpApi      = window.jsforce.require('./http-api');\n\n/*--------------------------------------------*/\n\n/**\n * Class for Bulk API Job\n *\n * @protected\n * @class Bulk~Job\n * @extends events.EventEmitter\n *\n * @param {Bulk} bulk - Bulk API object\n * @param {String} [type] - SObject type\n * @param {String} [operation] - Bulk load operation ('insert', 'update', 'upsert', 'delete', or 'hardDelete')\n * @param {Object} [options] - Options for bulk loading operation\n * @param {String} [options.extIdField] - External ID field name (used when upsert operation).\n * @param {String} [options.concurrencyMode] - 'Serial' or 'Parallel'. Defaults to Parallel.\n * @param {String} [jobId] - Job ID (if already available)\n */\nvar Job = function(bulk, type, operation, options, jobId) {\n  this._bulk = bulk;\n  this.type = type;\n  this.operation = operation;\n  this.options = options || {};\n  this.id = jobId;\n  this.state = this.id ? 'Open' : 'Unknown';\n  this._batches = {};\n};\n\ninherits(Job, events.EventEmitter);\n\n/**\n * @typedef {Object} Bulk~JobInfo\n * @prop {String} id - Job ID\n * @prop {String} object - Object type name\n * @prop {String} operation - Operation type of the job\n * @prop {String} state - Job status\n */\n\n/**\n * Return latest jobInfo from cache\n *\n * @method Bulk~Job#info\n * @param {Callback.<Bulk~JobInfo>} [callback] - Callback function\n * @returns {Promise.<Bulk~JobInfo>}\n */\nJob.prototype.info = function(callback) {\n  var self = this;\n  // if cache is not available, check the latest\n  if (!this._jobInfo) {\n    this._jobInfo = this.check();\n  }\n  return this._jobInfo.thenCall(callback);\n};\n\n/**\n * Open new job and get jobinfo\n *\n * @method Bulk~Job#open\n * @param {Callback.<Bulk~JobInfo>} [callback] - Callback function\n * @returns {Promise.<Bulk~JobInfo>}\n */\nJob.prototype.open = function(callback) {\n  var self = this;\n  var bulk = this._bulk;\n  var logger = bulk._logger;\n\n  // if not requested opening job\n  if (!this._jobInfo) {\n    var operation = this.operation.toLowerCase();\n    if (operation === 'harddelete') { operation = 'hardDelete'; }\n    var body = [\n      '<?xml version=\"1.0\" encoding=\"UTF-8\"?>',\n      '<jobInfo  xmlns=\"http://www.force.com/2009/06/asyncapi/dataload\">',\n        '<operation>' + operation + '</operation>',\n        '<object>' + this.type + '</object>',\n        (this.options.extIdField ?\n         '<externalIdFieldName>'+this.options.extIdField+'</externalIdFieldName>' :\n         ''),\n        (this.options.concurrencyMode ?\n         '<concurrencyMode>'+this.options.concurrencyMode+'</concurrencyMode>' :\n         ''),\n        (this.options.assignmentRuleId ?\n          '<assignmentRuleId>' + this.options.assignmentRuleId + '</assignmentRuleId>' :\n          ''),\n        '<contentType>CSV</contentType>',\n      '</jobInfo>'\n    ].join('');\n\n    this._jobInfo = bulk._request({\n      method : 'POST',\n      path : \"/job\",\n      body : body,\n      headers : {\n        \"Content-Type\" : \"application/xml; charset=utf-8\"\n      },\n      responseType: \"application/xml\"\n    }).then(function(res) {\n      self.emit(\"open\", res.jobInfo);\n      self.id = res.jobInfo.id;\n      self.state = res.jobInfo.state;\n      return res.jobInfo;\n    }, function(err) {\n      self.emit(\"error\", err);\n      throw err;\n    });\n  }\n  return this._jobInfo.thenCall(callback);\n};\n\n/**\n * Create a new batch instance in the job\n *\n * @method Bulk~Job#createBatch\n * @returns {Bulk~Batch}\n */\nJob.prototype.createBatch = function() {\n  var batch = new Batch(this);\n  var self = this;\n  batch.on('queue', function() {\n    self._batches[batch.id] = batch;\n  });\n  return batch;\n};\n\n/**\n * Get a batch instance specified by given batch ID\n *\n * @method Bulk~Job#batch\n * @param {String} batchId - Batch ID\n * @returns {Bulk~Batch}\n */\nJob.prototype.batch = function(batchId) {\n  var batch = this._batches[batchId];\n  if (!batch) {\n    batch = new Batch(this, batchId);\n    this._batches[batchId] = batch;\n  }\n  return batch;\n};\n\n/**\n * Check the latest job status from server\n *\n * @method Bulk~Job#check\n * @param {Callback.<Bulk~JobInfo>} [callback] - Callback function\n * @returns {Promise.<Bulk~JobInfo>}\n */\nJob.prototype.check = function(callback) {\n  var self = this;\n  var bulk = this._bulk;\n  var logger = bulk._logger;\n\n  this._jobInfo = this._waitAssign().then(function() {\n    return bulk._request({\n      method : 'GET',\n      path : \"/job/\" + self.id,\n      responseType: \"application/xml\"\n    });\n  }).then(function(res) {\n    logger.debug(res.jobInfo);\n    self.id = res.jobInfo.id;\n    self.type = res.jobInfo.object;\n    self.operation = res.jobInfo.operation;\n    self.state = res.jobInfo.state;\n    return res.jobInfo;\n  });\n  return this._jobInfo.thenCall(callback);\n};\n\n/**\n * Wait till the job is assigned to server\n *\n * @method Bulk~Job#info\n * @param {Callback.<Bulk~JobInfo>} [callback] - Callback function\n * @returns {Promise.<Bulk~JobInfo>}\n */\nJob.prototype._waitAssign = function(callback) {\n  return (this.id ? Promise.resolve({ id: this.id }) : this.open()).thenCall(callback);\n};\n\n\n/**\n * List all registered batch info in job\n *\n * @method Bulk~Job#list\n * @param {Callback.<Array.<Bulk~BatchInfo>>} [callback] - Callback function\n * @returns {Promise.<Array.<Bulk~BatchInfo>>}\n */\nJob.prototype.list = function(callback) {\n  var self = this;\n  var bulk = this._bulk;\n  var logger = bulk._logger;\n\n  return this._waitAssign().then(function() {\n    return bulk._request({\n      method : 'GET',\n      path : \"/job/\" + self.id + \"/batch\",\n      responseType: \"application/xml\"\n    });\n  }).then(function(res) {\n    logger.debug(res.batchInfoList.batchInfo);\n    var batchInfoList = res.batchInfoList;\n    batchInfoList = _.isArray(batchInfoList.batchInfo) ? batchInfoList.batchInfo : [ batchInfoList.batchInfo ];\n    return batchInfoList;\n  }).thenCall(callback);\n\n};\n\n/**\n * Close opened job\n *\n * @method Bulk~Job#close\n * @param {Callback.<Bulk~JobInfo>} [callback] - Callback function\n * @returns {Promise.<Bulk~JobInfo>}\n */\nJob.prototype.close = function() {\n  var self = this;\n  return this._changeState(\"Closed\").then(function(jobInfo) {\n    self.id = null;\n    self.emit(\"close\", jobInfo);\n    return jobInfo;\n  }, function(err) {\n    self.emit(\"error\", err);\n    throw err;\n  });\n};\n\n/**\n * Set the status to abort\n *\n * @method Bulk~Job#abort\n * @param {Callback.<Bulk~JobInfo>} [callback] - Callback function\n * @returns {Promise.<Bulk~JobInfo>}\n */\nJob.prototype.abort = function() {\n  var self = this;\n  return this._changeState(\"Aborted\").then(function(jobInfo) {\n    self.id = null;\n    self.emit(\"abort\", jobInfo);\n    return jobInfo;\n  }, function(err) {\n    self.emit(\"error\", err);\n    throw err;\n  });\n};\n\n/**\n * @private\n */\nJob.prototype._changeState = function(state, callback) {\n  var self = this;\n  var bulk = this._bulk;\n  var logger = bulk._logger;\n\n  this._jobInfo = this._waitAssign().then(function() {\n    var body = [\n      '<?xml version=\"1.0\" encoding=\"UTF-8\"?>',\n      '<jobInfo xmlns=\"http://www.force.com/2009/06/asyncapi/dataload\">',\n        '<state>' + state + '</state>',\n      '</jobInfo>'\n    ].join('');\n    return bulk._request({\n      method : 'POST',\n      path : \"/job/\" + self.id,\n      body : body,\n      headers : {\n        \"Content-Type\" : \"application/xml; charset=utf-8\"\n      },\n      responseType: \"application/xml\"\n    });\n  }).then(function(res) {\n    logger.debug(res.jobInfo);\n    self.state = res.jobInfo.state;\n    return res.jobInfo;\n  });\n  return this._jobInfo.thenCall(callback);\n\n};\n\n\n/*--------------------------------------------*/\n\n/**\n * Batch (extends RecordStream)\n *\n * @protected\n * @class Bulk~Batch\n * @extends {stream.Writable}\n * @implements {Promise.<Array.<RecordResult>>}\n * @param {Bulk~Job} job - Bulk job object\n * @param {String} [batchId] - Batch ID (if already available)\n */\nvar Batch = function(job, batchId) {\n  Batch.super_.call(this, { objectMode: true });\n  this.job = job;\n  this.id = batchId;\n  this._bulk = job._bulk;\n  this._deferred = Promise.defer();\n  this._setupDataStreams();\n};\n\ninherits(Batch, stream.Writable);\n\n\n/**\n * @private\n */\nBatch.prototype._setupDataStreams = function() {\n  var batch = this;\n  var converterOptions = { nullValue : '#N/A' };\n  this._uploadStream = new RecordStream.Serializable();\n  this._uploadDataStream = this._uploadStream.stream('csv', converterOptions);\n  this._downloadStream = new RecordStream.Parsable();\n  this._downloadDataStream = this._downloadStream.stream('csv', converterOptions);\n\n  this.on('finish', function() {\n    batch._uploadStream.end();\n  });\n  this._uploadDataStream.once('readable', function() {\n    batch.job.open().then(function() {\n      // pipe upload data to batch API request stream\n      batch._uploadDataStream.pipe(batch._createRequestStream());\n    });\n  });\n\n  // duplex data stream, opened access to API programmers by Batch#stream()\n  var dataStream = this._dataStream = new Duplex();\n  dataStream._write = function(data, enc, cb) {\n    batch._uploadDataStream.write(data, enc, cb);\n  };\n  dataStream.on('finish', function() {\n    batch._uploadDataStream.end();\n  });\n\n  this._downloadDataStream.on('readable', function() {\n    dataStream.read(0);\n  });\n  this._downloadDataStream.on('end', function() {\n    dataStream.push(null);\n  });\n  dataStream._read = function(size) {\n    var chunk;\n    while ((chunk = batch._downloadDataStream.read()) !== null) {\n      dataStream.push(chunk);\n    }\n  };\n};\n\n/**\n * Connect batch API and create stream instance of request/response\n *\n * @private\n * @returns {stream.Duplex}\n */\nBatch.prototype._createRequestStream = function() {\n  var batch = this;\n  var bulk = batch._bulk;\n  var logger = bulk._logger;\n\n  return bulk._request({\n    method : 'POST',\n    path : \"/job/\" + batch.job.id + \"/batch\",\n    headers: {\n      \"Content-Type\": \"text/csv\"\n    },\n    responseType: \"application/xml\"\n  }, function(err, res) {\n    if (err) {\n      batch.emit('error', err);\n    } else {\n      logger.debug(res.batchInfo);\n      batch.id = res.batchInfo.id;\n      batch.emit('queue', res.batchInfo);\n    }\n  }).stream();\n};\n\n/**\n * Implementation of Writable\n *\n * @override\n * @private\n */\nBatch.prototype._write = function(record, enc, cb) {\n  record = _.clone(record);\n  if (this.job.operation === \"insert\") {\n    delete record.Id;\n  } else if (this.job.operation === \"delete\") {\n    record = { Id: record.Id };\n  }\n  delete record.type;\n  delete record.attributes;\n  this._uploadStream.write(record, enc, cb);\n};\n\n/**\n * Returns duplex stream which accepts CSV data input and batch result output\n *\n * @returns {stream.Duplex}\n */\nBatch.prototype.stream = function() {\n  return this._dataStream;\n};\n\n/**\n * Execute batch operation\n *\n * @method Bulk~Batch#execute\n * @param {Array.<Record>|stream.Stream|String} [input] - Input source for batch operation. Accepts array of records, CSV string, and CSV data input stream in insert/update/upsert/delete/hardDelete operation, SOQL string in query operation.\n * @param {Callback.<Array.<RecordResult>|Array.<BatchResultInfo>>} [callback] - Callback function\n * @returns {Bulk~Batch}\n */\nBatch.prototype.run =\nBatch.prototype.exec =\nBatch.prototype.execute = function(input, callback) {\n  var self = this;\n\n  if (typeof input === 'function') { // if input argument is omitted\n    callback = input;\n    input = null;\n  }\n\n  // if batch is already executed\n  if (this._result) {\n    throw new Error(\"Batch already executed.\");\n  }\n\n  var rdeferred = Promise.defer();\n  this._result = rdeferred.promise;\n  this._result.then(function(res) {\n    self._deferred.resolve(res);\n  }, function(err) {\n    self._deferred.reject(err);\n  });\n  this.once('response', function(res) {\n    rdeferred.resolve(res);\n  });\n  this.once('error', function(err) {\n    rdeferred.reject(err);\n  });\n\n  if (_.isObject(input) && _.isFunction(input.pipe)) { // if input has stream.Readable interface\n    input.pipe(this._dataStream);\n  } else {\n    var data;\n    if (_.isArray(input)) {\n      _.forEach(input, function(record) {\n        Object.keys(record).forEach(function(key) {\n          if (typeof record[key] === 'boolean') {\n            record[key] = String(record[key])\n          }\n        })\n        self.write(record);\n      });\n      self.end();\n    } else if (_.isString(input)){\n      data = input;\n      this._dataStream.write(data, 'utf8');\n      this._dataStream.end();\n    }\n  }\n\n  // return Batch instance for chaining\n  return this.thenCall(callback);\n};\n\n/**\n * Promise/A+ interface\n * http://promises-aplus.github.io/promises-spec/\n *\n * Delegate to deferred promise, return promise instance for batch result\n *\n * @method Bulk~Batch#then\n */\nBatch.prototype.then = function(onResolved, onReject, onProgress) {\n  return this._deferred.promise.then(onResolved, onReject, onProgress);\n};\n\n/**\n * Promise/A+ extension\n * Call \"then\" using given node-style callback function\n *\n * @method Bulk~Batch#thenCall\n */\nBatch.prototype.thenCall = function(callback) {\n  if (_.isFunction(callback)) {\n    this.then(function(res) {\n      process.nextTick(function() {\n        callback(null, res);\n      });\n    }, function(err) {\n      process.nextTick(function() {\n        callback(err);\n      });\n    });\n  }\n  return this;\n};\n\n/**\n * @typedef {Object} Bulk~BatchInfo\n * @prop {String} id - Batch ID\n * @prop {String} jobId - Job ID\n * @prop {String} state - Batch state\n * @prop {String} stateMessage - Batch state message\n */\n\n/**\n * Check the latest batch status in server\n *\n * @method Bulk~Batch#check\n * @param {Callback.<Bulk~BatchInfo>} [callback] - Callback function\n * @returns {Promise.<Bulk~BatchInfo>}\n */\nBatch.prototype.check = function(callback) {\n  var self = this;\n  var bulk = this._bulk;\n  var logger = bulk._logger;\n  var jobId = this.job.id;\n  var batchId = this.id;\n\n  if (!jobId || !batchId) {\n    throw new Error(\"Batch not started.\");\n  }\n  return bulk._request({\n    method : 'GET',\n    path : \"/job/\" + jobId + \"/batch/\" + batchId,\n    responseType: \"application/xml\"\n  }).then(function(res) {\n    logger.debug(res.batchInfo);\n    return res.batchInfo;\n  }).thenCall(callback);\n};\n\n\n/**\n * Polling the batch result and retrieve\n *\n * @method Bulk~Batch#poll\n * @param {Number} interval - Polling interval in milliseconds\n * @param {Number} timeout - Polling timeout in milliseconds\n */\nBatch.prototype.poll = function(interval, timeout) {\n  var self = this;\n  var jobId = this.job.id;\n  var batchId = this.id;\n\n  if (!jobId || !batchId) {\n    throw new Error(\"Batch not started.\");\n  }\n  var startTime = new Date().getTime();\n  var poll = function() {\n    var now = new Date().getTime();\n    if (startTime + timeout < now) {\n      var err = new Error(\"Polling time out. Job Id = \" + jobId + \" , batch Id = \" + batchId);\n      err.name = 'PollingTimeout';\n      err.jobId = jobId;\n      err.batchId = batchId;\n      self.emit('error', err);\n      return;\n    }\n    self.check(function(err, res) {\n      if (err) {\n        self.emit('error', err);\n      } else {\n        if (res.state === \"Failed\") {\n          if (parseInt(res.numberRecordsProcessed, 10) > 0) {\n            self.retrieve();\n          } else {\n            self.emit('error', new Error(res.stateMessage));\n          }\n        } else if (res.state === \"Completed\") {\n          self.retrieve();\n        } else {\n          self.emit('progress', res);\n          setTimeout(poll, interval);\n        }\n      }\n    });\n  };\n  setTimeout(poll, interval);\n};\n\n/**\n * @typedef {Object} Bulk~BatchResultInfo\n * @prop {String} id - Batch result ID\n * @prop {String} batchId - Batch ID which includes this batch result.\n * @prop {String} jobId - Job ID which includes this batch result.\n */\n\n/**\n * Retrieve batch result\n *\n * @method Bulk~Batch#retrieve\n * @param {Callback.<Array.<RecordResult>|Array.<Bulk~BatchResultInfo>>} [callback] - Callback function\n * @returns {Promise.<Array.<RecordResult>|Array.<Bulk~BatchResultInfo>>}\n */\nBatch.prototype.retrieve = function(callback) {\n  var self = this;\n  var bulk = this._bulk;\n  var jobId = this.job.id;\n  var job = this.job;\n  var batchId = this.id;\n\n  if (!jobId || !batchId) {\n    throw new Error(\"Batch not started.\");\n  }\n\n  return job.info().then(function(jobInfo) {\n    return bulk._request({\n      method : 'GET',\n      path : \"/job/\" + jobId + \"/batch/\" + batchId + \"/result\"\n    });\n  }).then(function(res) {\n    var results;\n    if (job.operation === 'query') {\n      var conn = bulk._conn;\n      var resultIds = res['result-list'].result;\n      results = res['result-list'].result;\n      results = _.map(_.isArray(results) ? results : [ results ], function(id) {\n        return {\n          id: id,\n          batchId: batchId,\n          jobId: jobId\n        };\n      });\n    } else {\n      results = _.map(res, function(ret) {\n        return {\n          id: ret.Id || null,\n          success: ret.Success === \"true\",\n          errors: ret.Error ? [ ret.Error ] : []\n        };\n      });\n    }\n    self.emit('response', results);\n    return results;\n  }).fail(function(err) {\n    self.emit('error', err);\n    throw err;\n  }).thenCall(callback);\n};\n\n/**\n * Fetch query result as a record stream\n * @param {String} resultId - Result id\n * @returns {RecordStream} - Record stream, convertible to CSV data stream\n */\nBatch.prototype.result = function(resultId) {\n  var jobId = this.job.id;\n  var batchId = this.id;\n  if (!jobId || !batchId) {\n    throw new Error(\"Batch not started.\");\n  }\n  var resultStream = new RecordStream.Parsable();\n  var resultDataStream = resultStream.stream('csv');\n  var reqStream = this._bulk._request({\n    method : 'GET',\n    path : \"/job/\" + jobId + \"/batch/\" + batchId + \"/result/\" + resultId,\n    responseType: \"application/octet-stream\"\n  }).stream().pipe(resultDataStream);\n  return resultStream;\n};\n\n/*--------------------------------------------*/\n/**\n * @private\n */\nvar BulkApi = function() {\n  BulkApi.super_.apply(this, arguments);\n};\n\ninherits(BulkApi, HttpApi);\n\nBulkApi.prototype.beforeSend = function(request) {\n  request.headers = request.headers || {};\n  request.headers[\"X-SFDC-SESSION\"] = this._conn.accessToken;\n};\n\nBulkApi.prototype.isSessionExpired = function(response) {\n  return response.statusCode === 400 &&\n    /<exceptionCode>InvalidSessionId<\\/exceptionCode>/.test(response.body);\n};\n\nBulkApi.prototype.hasErrorInResponseBody = function(body) {\n  return !!body.error;\n};\n\nBulkApi.prototype.parseError = function(body) {\n  return {\n    errorCode: body.error.exceptionCode,\n    message: body.error.exceptionMessage\n  };\n};\n\n/*--------------------------------------------*/\n\n/**\n * Class for Bulk API\n *\n * @class\n * @param {Connection} conn - Connection object\n */\nvar Bulk = function(conn) {\n  this._conn = conn;\n  this._logger = conn._logger;\n};\n\n/**\n * Polling interval in milliseconds\n * @type {Number}\n */\nBulk.prototype.pollInterval = 1000;\n\n/**\n * Polling timeout in milliseconds\n * @type {Number}\n */\nBulk.prototype.pollTimeout = 10000;\n\n/** @private **/\nBulk.prototype._request = function(request, callback) {\n  var conn = this._conn;\n  request = _.clone(request);\n  var baseUrl = [ conn.instanceUrl, \"services/async\", conn.version ].join('/');\n  request.url = baseUrl + request.path;\n  var options = { responseType: request.responseType };\n  delete request.path;\n  delete request.responseType;\n  return new BulkApi(this._conn, options).request(request).thenCall(callback);\n};\n\n/**\n * Create and start bulkload job and batch\n *\n * @param {String} type - SObject type\n * @param {String} operation - Bulk load operation ('insert', 'update', 'upsert', 'delete', or 'hardDelete')\n * @param {Object} [options] - Options for bulk loading operation\n * @param {String} [options.extIdField] - External ID field name (used when upsert operation).\n * @param {String} [options.concurrencyMode] - 'Serial' or 'Parallel'. Defaults to Parallel.\n * @param {Array.<Record>|stream.Stream|String} [input] - Input source for bulkload. Accepts array of records, CSV string, and CSV data input stream in insert/update/upsert/delete/hardDelete operation, SOQL string in query operation.\n * @param {Callback.<Array.<RecordResult>|Array.<Bulk~BatchResultInfo>>} [callback] - Callback function\n * @returns {Bulk~Batch}\n */\nBulk.prototype.load = function(type, operation, options, input, callback) {\n  var self = this;\n  if (!type || !operation) {\n    throw new Error(\"Insufficient arguments. At least, 'type' and 'operation' are required.\");\n  }\n  if (!_.isObject(options) || options.constructor !== Object) { // when options is not plain hash object, it is omitted\n    callback = input;\n    input = options;\n    options = null;\n  }\n  var job = this.createJob(type, operation, options);\n  job.once('error', function (error) {\n    if (batch) {\n      batch.emit('error', error); // pass job error to batch\n    }\n  });\n  var batch = job.createBatch();\n  var cleanup = function() {\n    batch = null;\n    job.close();\n  };\n  var cleanupOnError = function(err) {\n    if (err.name !== 'PollingTimeout') {\n      cleanup();\n    }\n  };\n  batch.on('response', cleanup);\n  batch.on('error', cleanupOnError);\n  batch.on('queue', function() { batch.poll(self.pollInterval, self.pollTimeout); });\n  return batch.execute(input, callback);\n};\n\n/**\n * Execute bulk query and get record stream\n *\n * @param {String} soql - SOQL to execute in bulk job\n * @returns {RecordStream.Parsable} - Record stream, convertible to CSV data stream\n */\nBulk.prototype.query = function(soql) {\n  var m = soql.replace(/\\([\\s\\S]+\\)/g, '').match(/FROM\\s+(\\w+)/i);\n  if (!m) {\n    throw new Error(\"No sobject type found in query, maybe caused by invalid SOQL.\");\n  }\n  var type = m[1];\n  var self = this;\n  var recordStream = new RecordStream.Parsable();\n  var dataStream = recordStream.stream('csv');\n  this.load(type, \"query\", soql).then(function(results) {\n    var streams = results.map(function(result) {\n      return self\n        .job(result.jobId)\n        .batch(result.batchId)\n        .result(result.id)\n        .stream();\n    });\n\n    joinStreams(streams).pipe(dataStream);\n  }).fail(function(err) {\n    recordStream.emit('error', err);\n  });\n  return recordStream;\n};\n\n\n/**\n * Create a new job instance\n *\n * @param {String} type - SObject type\n * @param {String} operation - Bulk load operation ('insert', 'update', 'upsert', 'delete', 'hardDelete', or 'query')\n * @param {Object} [options] - Options for bulk loading operation\n * @returns {Bulk~Job}\n */\nBulk.prototype.createJob = function(type, operation, options) {\n  return new Job(this, type, operation, options);\n};\n\n/**\n * Get a job instance specified by given job ID\n *\n * @param {String} jobId - Job ID\n * @returns {Bulk~Job}\n */\nBulk.prototype.job = function(jobId) {\n  return new Job(this, null, null, null, jobId);\n};\n\n\n/*--------------------------------------------*/\n/*\n * Register hook in connection instantiation for dynamically adding this API module features\n */\njsforce.on('connection:new', function(conn) {\n  conn.bulk = new Bulk(conn);\n});\n\n\nmodule.exports = Bulk;\n","// shim for using process in browser\nvar process = module.exports = {};\n\n// cached from whatever global is present so that test runners that stub it\n// don't break things.  But we need to wrap it in a try catch in case it is\n// wrapped in strict mode code which doesn't define any globals.  It's inside a\n// function because try/catches deoptimize in certain engines.\n\nvar cachedSetTimeout;\nvar cachedClearTimeout;\n\nfunction defaultSetTimout() {\n    throw new Error('setTimeout has not been defined');\n}\nfunction defaultClearTimeout () {\n    throw new Error('clearTimeout has not been defined');\n}\n(function () {\n    try {\n        if (typeof setTimeout === 'function') {\n            cachedSetTimeout = setTimeout;\n        } else {\n            cachedSetTimeout = defaultSetTimout;\n        }\n    } catch (e) {\n        cachedSetTimeout = defaultSetTimout;\n    }\n    try {\n        if (typeof clearTimeout === 'function') {\n            cachedClearTimeout = clearTimeout;\n        } else {\n            cachedClearTimeout = defaultClearTimeout;\n        }\n    } catch (e) {\n        cachedClearTimeout = defaultClearTimeout;\n    }\n} ())\nfunction runTimeout(fun) {\n    if (cachedSetTimeout === setTimeout) {\n        //normal enviroments in sane situations\n        return setTimeout(fun, 0);\n    }\n    // if setTimeout wasn't available but was latter defined\n    if ((cachedSetTimeout === defaultSetTimout || !cachedSetTimeout) && setTimeout) {\n        cachedSetTimeout = setTimeout;\n        return setTimeout(fun, 0);\n    }\n    try {\n        // when when somebody has screwed with setTimeout but no I.E. maddness\n        return cachedSetTimeout(fun, 0);\n    } catch(e){\n        try {\n            // When we are in I.E. but the script has been evaled so I.E. doesn't trust the global object when called normally\n            return cachedSetTimeout.call(null, fun, 0);\n        } catch(e){\n            // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error\n            return cachedSetTimeout.call(this, fun, 0);\n        }\n    }\n\n\n}\nfunction runClearTimeout(marker) {\n    if (cachedClearTimeout === clearTimeout) {\n        //normal enviroments in sane situations\n        return clearTimeout(marker);\n    }\n    // if clearTimeout wasn't available but was latter defined\n    if ((cachedClearTimeout === defaultClearTimeout || !cachedClearTimeout) && clearTimeout) {\n        cachedClearTimeout = clearTimeout;\n        return clearTimeout(marker);\n    }\n    try {\n        // when when somebody has screwed with setTimeout but no I.E. maddness\n        return cachedClearTimeout(marker);\n    } catch (e){\n        try {\n            // When we are in I.E. but the script has been evaled so I.E. doesn't  trust the global object when called normally\n            return cachedClearTimeout.call(null, marker);\n        } catch (e){\n            // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error.\n            // Some versions of I.E. have different rules for clearTimeout vs setTimeout\n            return cachedClearTimeout.call(this, marker);\n        }\n    }\n\n\n\n}\nvar queue = [];\nvar draining = false;\nvar currentQueue;\nvar queueIndex = -1;\n\nfunction cleanUpNextTick() {\n    if (!draining || !currentQueue) {\n        return;\n    }\n    draining = false;\n    if (currentQueue.length) {\n        queue = currentQueue.concat(queue);\n    } else {\n        queueIndex = -1;\n    }\n    if (queue.length) {\n        drainQueue();\n    }\n}\n\nfunction drainQueue() {\n    if (draining) {\n        return;\n    }\n    var timeout = runTimeout(cleanUpNextTick);\n    draining = true;\n\n    var len = queue.length;\n    while(len) {\n        currentQueue = queue;\n        queue = [];\n        while (++queueIndex < len) {\n            if (currentQueue) {\n                currentQueue[queueIndex].run();\n            }\n        }\n        queueIndex = -1;\n        len = queue.length;\n    }\n    currentQueue = null;\n    draining = false;\n    runClearTimeout(timeout);\n}\n\nprocess.nextTick = function (fun) {\n    var args = new Array(arguments.length - 1);\n    if (arguments.length > 1) {\n        for (var i = 1; i < arguments.length; i++) {\n            args[i - 1] = arguments[i];\n        }\n    }\n    queue.push(new Item(fun, args));\n    if (queue.length === 1 && !draining) {\n        runTimeout(drainQueue);\n    }\n};\n\n// v8 likes predictible objects\nfunction Item(fun, array) {\n    this.fun = fun;\n    this.array = array;\n}\nItem.prototype.run = function () {\n    this.fun.apply(null, this.array);\n};\nprocess.title = 'browser';\nprocess.browser = true;\nprocess.env = {};\nprocess.argv = [];\nprocess.version = ''; // empty string to avoid regexp issues\nprocess.versions = {};\n\nfunction noop() {}\n\nprocess.on = noop;\nprocess.addListener = noop;\nprocess.once = noop;\nprocess.off = noop;\nprocess.removeListener = noop;\nprocess.removeAllListeners = noop;\nprocess.emit = noop;\nprocess.prependListener = noop;\nprocess.prependOnceListener = noop;\n\nprocess.listeners = function (name) { return [] }\n\nprocess.binding = function (name) {\n    throw new Error('process.binding is not supported');\n};\n\nprocess.cwd = function () { return '/' };\nprocess.chdir = function (dir) {\n    throw new Error('process.chdir is not supported');\n};\nprocess.umask = function() { return 0; };\n"]}
|