GH-21 Support for binary attributes
This commit is contained in:
parent
9610cbe9d3
commit
c20030f5bb
|
@ -2,6 +2,8 @@
|
||||||
|
|
||||||
var assert = require('assert');
|
var assert = require('assert');
|
||||||
|
|
||||||
|
var asn1 = require('asn1');
|
||||||
|
|
||||||
var Protocol = require('./protocol');
|
var Protocol = require('./protocol');
|
||||||
|
|
||||||
|
|
||||||
|
@ -14,32 +16,69 @@ function Attribute(options) {
|
||||||
throw new TypeError('options must be an object');
|
throw new TypeError('options must be an object');
|
||||||
if (options.type && typeof(options.type) !== 'string')
|
if (options.type && typeof(options.type) !== 'string')
|
||||||
throw new TypeError('options.type must be a string');
|
throw new TypeError('options.type must be a string');
|
||||||
if (options.vals && !Array.isArray(options.vals))
|
|
||||||
throw new TypeErrr('options.vals must be an array[string]');
|
|
||||||
if (options.vals && options.vals.length) {
|
|
||||||
options.vals.forEach(function(v) {
|
|
||||||
if (typeof(v) !== 'string')
|
|
||||||
throw new TypeErrr('options.vals must be an array[string]');
|
|
||||||
});
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
options = {};
|
options = {};
|
||||||
}
|
}
|
||||||
|
|
||||||
this.type = options.type || '';
|
|
||||||
this.vals = options.vals ? options.vals.slice(0) : [];
|
|
||||||
|
|
||||||
var self = this;
|
var self = this;
|
||||||
|
|
||||||
|
this.type = options.type || '';
|
||||||
|
this._vals = [];
|
||||||
|
|
||||||
|
this.__defineGetter__('vals', function() {
|
||||||
|
var _vals = [];
|
||||||
|
|
||||||
|
self._vals.forEach(function(v) {
|
||||||
|
if (/;binary$/.test(self.type)) {
|
||||||
|
_vals.push(v.toString('base64'));
|
||||||
|
} else {
|
||||||
|
_vals.push(v.toString('utf8'));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return _vals;
|
||||||
|
});
|
||||||
|
|
||||||
|
this.__defineSetter__('vals', function(vals) {
|
||||||
|
if (Array.isArray(vals)) {
|
||||||
|
vals.forEach(function(v) {
|
||||||
|
self.addValue(v);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
self.addValue(vals);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
this.__defineGetter__('buffers', function() {
|
||||||
|
return self._vals;
|
||||||
|
});
|
||||||
|
|
||||||
this.__defineGetter__('json', function() {
|
this.__defineGetter__('json', function() {
|
||||||
return {
|
return {
|
||||||
type: self.type,
|
type: self.type,
|
||||||
vals: self.vals
|
vals: self.vals
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
|
if (options.vals)
|
||||||
|
this.vals = options.vals;
|
||||||
|
|
||||||
}
|
}
|
||||||
module.exports = Attribute;
|
module.exports = Attribute;
|
||||||
|
|
||||||
|
|
||||||
|
Attribute.prototype.addValue = function(val) {
|
||||||
|
if (Buffer.isBuffer(val)) {
|
||||||
|
this._vals.push(val);
|
||||||
|
} else {
|
||||||
|
var encoding = 'utf8';
|
||||||
|
if (/;binary$/.test(this.type))
|
||||||
|
encoding = 'base64';
|
||||||
|
this._vals.push(new Buffer(val + '', encoding));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
Attribute.compare = function(a, b) {
|
Attribute.compare = function(a, b) {
|
||||||
if (!(a instanceof Attribute) || !(b instanceof Attribute))
|
if (!(a instanceof Attribute) || !(b instanceof Attribute))
|
||||||
throw new TypeError('can only compare Attributes');
|
throw new TypeError('can only compare Attributes');
|
||||||
|
@ -57,13 +96,6 @@ Attribute.compare = function(a, b) {
|
||||||
return 0;
|
return 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
Attribute.prototype.addValue = function(val) {
|
|
||||||
if (typeof(val) !== 'string')
|
|
||||||
throw new TypeError('val (string) required');
|
|
||||||
|
|
||||||
this.vals.push(val);
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
Attribute.prototype.parse = function(ber) {
|
Attribute.prototype.parse = function(ber) {
|
||||||
assert.ok(ber);
|
assert.ok(ber);
|
||||||
|
@ -75,7 +107,7 @@ Attribute.prototype.parse = function(ber) {
|
||||||
if (ber.readSequence(Protocol.LBER_SET)) {
|
if (ber.readSequence(Protocol.LBER_SET)) {
|
||||||
var end = ber.offset + ber.length;
|
var end = ber.offset + ber.length;
|
||||||
while (ber.offset < end)
|
while (ber.offset < end)
|
||||||
this.vals.push(ber.readString());
|
this._vals.push(ber.readString(asn1.Ber.OctetString, true));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -86,10 +118,15 @@ Attribute.prototype.parse = function(ber) {
|
||||||
Attribute.prototype.toBer = function(ber) {
|
Attribute.prototype.toBer = function(ber) {
|
||||||
assert.ok(ber);
|
assert.ok(ber);
|
||||||
|
|
||||||
|
var vals = [];
|
||||||
|
this._vals.forEach(function(v) {
|
||||||
|
vals.push(v.toString('utf8'));
|
||||||
|
});
|
||||||
|
|
||||||
ber.startSequence();
|
ber.startSequence();
|
||||||
ber.writeString(this.type);
|
ber.writeString(this.type);
|
||||||
ber.startSequence(Protocol.LBER_SET);
|
ber.startSequence(Protocol.LBER_SET);
|
||||||
ber.writeStringArray(this.vals && this.vals.length ? this.vals : []);
|
ber.writeStringArray(vals);
|
||||||
ber.endSequence();
|
ber.endSequence();
|
||||||
ber.endSequence();
|
ber.endSequence();
|
||||||
|
|
||||||
|
@ -108,7 +145,8 @@ Attribute.isAttribute = function(attr) {
|
||||||
if (!attr.type || typeof(attr.type) !== 'string') return false;
|
if (!attr.type || typeof(attr.type) !== 'string') return false;
|
||||||
if (!attr.vals || !Array.isArray(attr.vals)) return false;
|
if (!attr.vals || !Array.isArray(attr.vals)) return false;
|
||||||
for (var i = 0; i < attr.vals.length; i++)
|
for (var i = 0; i < attr.vals.length; i++)
|
||||||
if (typeof(attr.vals[i]) !== 'string') return false;
|
if (typeof(attr.vals[i]) !== 'string' && !Buffer.isBuffer(attr.vals[i]))
|
||||||
|
return false;
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
};
|
};
|
||||||
|
|
|
@ -87,20 +87,7 @@ SearchEntry.prototype.fromObject = function(obj) {
|
||||||
this.attributes = [];
|
this.attributes = [];
|
||||||
|
|
||||||
Object.keys(obj).forEach(function(k) {
|
Object.keys(obj).forEach(function(k) {
|
||||||
var attr = new Attribute({type: k, vals: []});
|
self.attributes.push(new Attribute({type: k, vals: obj[k]}));
|
||||||
|
|
||||||
if (Array.isArray(obj[k])) {
|
|
||||||
obj[k].forEach(function(v) {
|
|
||||||
if (typeof(v) !== 'string')
|
|
||||||
v = '' + v;
|
|
||||||
|
|
||||||
attr.vals.push(v);
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
attr.vals.push(obj[k] + '');
|
|
||||||
}
|
|
||||||
|
|
||||||
self.attributes.push(attr);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|
|
@ -13,7 +13,7 @@
|
||||||
"node": ">=0.4"
|
"node": ">=0.4"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"asn1": "~0.1.5",
|
"asn1": "~0.1.6",
|
||||||
"buffertools": "~1.0.3",
|
"buffertools": "~1.0.3",
|
||||||
"dtrace-provider": "~0.0.3",
|
"dtrace-provider": "~0.0.3",
|
||||||
"sprintf": "~0.1.1"
|
"sprintf": "~0.1.1"
|
||||||
|
|
|
@ -83,7 +83,8 @@ test('setup', function(t) {
|
||||||
res.send(res.createSearchEntry({
|
res.send(res.createSearchEntry({
|
||||||
objectName: req.dn,
|
objectName: req.dn,
|
||||||
attributes: {
|
attributes: {
|
||||||
'foo;binary': '\u00bd + \u00bc = \u00be'
|
'foo;binary': 'wr0gKyDCvCA9IMK+',
|
||||||
|
'objectclass': 'binary'
|
||||||
}
|
}
|
||||||
}));
|
}));
|
||||||
} else {
|
} else {
|
||||||
|
@ -431,6 +432,39 @@ test('search empty attribute', function(t) {
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
|
test('GH-21 binary attributes', function(t) {
|
||||||
|
client.search('cn=bin, ' + SUFFIX, '(objectclass=*)', function(err, res) {
|
||||||
|
t.ifError(err);
|
||||||
|
t.ok(res);
|
||||||
|
var gotEntry = 0;
|
||||||
|
var expect = new Buffer('\u00bd + \u00bc = \u00be', 'utf8');
|
||||||
|
res.on('searchEntry', function(entry) {
|
||||||
|
t.ok(entry);
|
||||||
|
t.ok(entry instanceof ldap.SearchEntry);
|
||||||
|
t.equal(entry.dn.toString(), 'cn=bin, ' + SUFFIX);
|
||||||
|
t.ok(entry.attributes);
|
||||||
|
t.ok(entry.attributes.length);
|
||||||
|
t.equal(entry.attributes[0].type, 'foo;binary');
|
||||||
|
t.equal(entry.attributes[0].vals[0], expect.toString('base64'));
|
||||||
|
t.equal(entry.attributes[0].buffers[0].toString('base64'),
|
||||||
|
expect.toString('base64'));
|
||||||
|
t.ok(entry.object);
|
||||||
|
gotEntry++;
|
||||||
|
});
|
||||||
|
res.on('error', function(err) {
|
||||||
|
t.fail(err);
|
||||||
|
});
|
||||||
|
res.on('end', function(res) {
|
||||||
|
t.ok(res);
|
||||||
|
t.ok(res instanceof ldap.SearchResponse);
|
||||||
|
t.equal(res.status, 0);
|
||||||
|
t.equal(gotEntry, 1);
|
||||||
|
t.end();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
test('GH-23 case insensitive attribute filtering', function(t) {
|
test('GH-23 case insensitive attribute filtering', function(t) {
|
||||||
var opts = {
|
var opts = {
|
||||||
filter: '(objectclass=*)',
|
filter: '(objectclass=*)',
|
||||||
|
|
Loading…
Reference in New Issue