GH-21 Support for binary attributes
This commit is contained in:
parent
9610cbe9d3
commit
c20030f5bb
|
@ -2,6 +2,8 @@
|
|||
|
||||
var assert = require('assert');
|
||||
|
||||
var asn1 = require('asn1');
|
||||
|
||||
var Protocol = require('./protocol');
|
||||
|
||||
|
||||
|
@ -14,32 +16,69 @@ function Attribute(options) {
|
|||
throw new TypeError('options must be an object');
|
||||
if (options.type && typeof(options.type) !== '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 {
|
||||
options = {};
|
||||
}
|
||||
|
||||
this.type = options.type || '';
|
||||
this.vals = options.vals ? options.vals.slice(0) : [];
|
||||
|
||||
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() {
|
||||
return {
|
||||
type: self.type,
|
||||
vals: self.vals
|
||||
};
|
||||
});
|
||||
|
||||
if (options.vals)
|
||||
this.vals = options.vals;
|
||||
|
||||
}
|
||||
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) {
|
||||
if (!(a instanceof Attribute) || !(b instanceof Attribute))
|
||||
throw new TypeError('can only compare Attributes');
|
||||
|
@ -57,13 +96,6 @@ Attribute.compare = function(a, b) {
|
|||
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) {
|
||||
assert.ok(ber);
|
||||
|
@ -75,7 +107,7 @@ Attribute.prototype.parse = function(ber) {
|
|||
if (ber.readSequence(Protocol.LBER_SET)) {
|
||||
var end = ber.offset + ber.length;
|
||||
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) {
|
||||
assert.ok(ber);
|
||||
|
||||
var vals = [];
|
||||
this._vals.forEach(function(v) {
|
||||
vals.push(v.toString('utf8'));
|
||||
});
|
||||
|
||||
ber.startSequence();
|
||||
ber.writeString(this.type);
|
||||
ber.startSequence(Protocol.LBER_SET);
|
||||
ber.writeStringArray(this.vals && this.vals.length ? this.vals : []);
|
||||
ber.writeStringArray(vals);
|
||||
ber.endSequence();
|
||||
ber.endSequence();
|
||||
|
||||
|
@ -108,7 +145,8 @@ Attribute.isAttribute = function(attr) {
|
|||
if (!attr.type || typeof(attr.type) !== 'string') return false;
|
||||
if (!attr.vals || !Array.isArray(attr.vals)) return false;
|
||||
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;
|
||||
};
|
||||
|
|
|
@ -87,20 +87,7 @@ SearchEntry.prototype.fromObject = function(obj) {
|
|||
this.attributes = [];
|
||||
|
||||
Object.keys(obj).forEach(function(k) {
|
||||
var attr = new Attribute({type: k, vals: []});
|
||||
|
||||
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);
|
||||
self.attributes.push(new Attribute({type: k, vals: obj[k]}));
|
||||
});
|
||||
|
||||
return true;
|
||||
|
|
|
@ -13,7 +13,7 @@
|
|||
"node": ">=0.4"
|
||||
},
|
||||
"dependencies": {
|
||||
"asn1": "~0.1.5",
|
||||
"asn1": "~0.1.6",
|
||||
"buffertools": "~1.0.3",
|
||||
"dtrace-provider": "~0.0.3",
|
||||
"sprintf": "~0.1.1"
|
||||
|
|
|
@ -83,7 +83,8 @@ test('setup', function(t) {
|
|||
res.send(res.createSearchEntry({
|
||||
objectName: req.dn,
|
||||
attributes: {
|
||||
'foo;binary': '\u00bd + \u00bc = \u00be'
|
||||
'foo;binary': 'wr0gKyDCvCA9IMK+',
|
||||
'objectclass': 'binary'
|
||||
}
|
||||
}));
|
||||
} 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) {
|
||||
var opts = {
|
||||
filter: '(objectclass=*)',
|
||||
|
|
Loading…
Reference in New Issue