kafka-node
Version:
Client for Apache Kafka v0.9.x, v0.10.x and v0.11.x
126 lines (111 loc) • 3.45 kB
JavaScript
var util = require('util');
var async = require('async');
var EventEmitter = require('events');
function Offset (client) {
EventEmitter.call(this);
var self = this;
this.client = client;
this.ready = this.client.ready;
this.client.on('ready', function () {
self.ready = true;
self.emit('ready');
});
this.client.once('connect', function () {
self.emit('connect');
});
this.client.on('error', function (err) {
self.emit('error', err);
});
}
util.inherits(Offset, EventEmitter);
Offset.prototype.fetch = function (payloads, cb) {
if (!this.ready) {
this.once('ready', () => this.fetch(payloads, cb));
return;
}
this.client.sendOffsetRequest(this.buildPayloads(payloads), cb);
};
Offset.prototype.buildPayloads = function (payloads) {
return payloads.map(function (p) {
p.partition = p.partition || 0;
p.time = p.time || Date.now();
p.maxNum = p.maxNum || 1;
p.metadata = 'm'; // metadata can be arbitrary
return p;
});
};
Offset.prototype.buildOffsetFetchV1Payloads = function (payloads) {
return payloads.reduce(function (out, p) {
out[p.topic] = out[p.topic] || [];
out[p.topic].push(p.partition || 0);
return out;
}, {});
};
Offset.prototype.commit = function (groupId, payloads, cb) {
if (!this.ready) {
this.once('ready', () => this.commit(groupId, payloads, cb));
return;
}
this.client.sendOffsetCommitRequest(groupId, this.buildPayloads(payloads), cb);
};
Offset.prototype.fetchCommits = Offset.prototype.fetchCommitsV1 = function (groupId, payloads, cb) {
if (!this.ready) {
this.once('ready', () => this.fetchCommitsV1(groupId, payloads, cb));
return;
}
this.client.setCoordinatorIdAndSendOffsetFetchV1Request(groupId, this.buildOffsetFetchV1Payloads(payloads), cb);
};
Offset.prototype.fetchLatestOffsets = function (topics, cb) {
fetchOffsets(this, topics, cb, -1);
};
Offset.prototype.fetchEarliestOffsets = function (topics, cb) {
fetchOffsets(this, topics, cb, -2);
};
// private helper
function fetchOffsets (offset, topics, cb, when) {
if (!offset.ready) {
if (when === -1) {
offset.once('ready', () => offset.fetchLatestOffsets(topics, cb));
} else if (when === -2) {
offset.once('ready', () => offset.fetchEarliestOffsets(topics, cb));
}
return;
}
async.waterfall(
[
callback => {
offset.client.loadMetadataForTopics(topics, callback);
},
(topicsMetaData, callback) => {
var payloads = [];
var metaDatas = topicsMetaData[1].metadata;
Object.keys(metaDatas).forEach(function (topicName) {
var topic = metaDatas[topicName];
Object.keys(topic).forEach(function (partition) {
payloads.push({
topic: topicName,
partition: partition,
time: when
});
});
});
if (payloads.length === 0) {
return callback(new Error('Topic(s) does not exist'));
}
offset.fetch(payloads, callback);
},
function (results, callback) {
Object.keys(results).forEach(function (topicName) {
var topic = results[topicName];
Object.keys(topic).forEach(function (partitionName) {
topic[partitionName] = topic[partitionName][0];
});
});
callback(null, results);
}
],
cb
);
}
module.exports = Offset;
;