2011-08-04 20:32:01 +00:00
|
|
|
// Copyright 2011 Mark Cavage, Inc. All rights reserved.
|
|
|
|
|
|
|
|
var assert = require('assert');
|
|
|
|
var util = require('util');
|
|
|
|
|
|
|
|
var LDAPResult = require('./result');
|
|
|
|
var SearchEntry = require('./search_entry');
|
2011-09-27 18:49:33 +00:00
|
|
|
var SearchReference = require('./search_reference');
|
2011-08-04 20:32:01 +00:00
|
|
|
|
2011-11-15 18:49:23 +00:00
|
|
|
var dtrace = require('../dtrace');
|
2011-08-12 23:37:47 +00:00
|
|
|
var parseDN = require('../dn').parse;
|
2011-09-27 18:49:33 +00:00
|
|
|
var parseURL = require('../url').parse;
|
2011-08-04 20:32:01 +00:00
|
|
|
var Protocol = require('../protocol');
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
///--- API
|
|
|
|
|
|
|
|
function SearchResponse(options) {
|
|
|
|
if (!options)
|
|
|
|
options = {};
|
2012-02-18 08:15:52 +00:00
|
|
|
if (typeof (options) !== 'object')
|
2011-08-04 20:32:01 +00:00
|
|
|
throw new TypeError('options must be an object');
|
|
|
|
|
|
|
|
options.protocolOp = Protocol.LDAP_REP_SEARCH;
|
|
|
|
LDAPResult.call(this, options);
|
2011-08-12 23:37:47 +00:00
|
|
|
|
|
|
|
this.attributes = options.attributes ? options.attributes.slice() : [];
|
|
|
|
this.notAttributes = [];
|
2011-09-13 21:49:21 +00:00
|
|
|
this.sentEntries = 0;
|
2011-08-04 20:32:01 +00:00
|
|
|
}
|
|
|
|
util.inherits(SearchResponse, LDAPResult);
|
|
|
|
module.exports = SearchResponse;
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Allows you to send a SearchEntry back to the client.
|
|
|
|
*
|
|
|
|
* @param {Object} entry an instance of SearchEntry.
|
2011-09-16 16:06:35 +00:00
|
|
|
* @param {Boolean} nofiltering skip filtering notAttributes and '_' attributes.
|
|
|
|
* Defaults to 'false'.
|
2011-08-04 20:32:01 +00:00
|
|
|
*/
|
2012-02-18 08:15:52 +00:00
|
|
|
SearchResponse.prototype.send = function (entry, nofiltering) {
|
|
|
|
if (!entry || typeof (entry) !== 'object')
|
2011-08-04 20:32:01 +00:00
|
|
|
throw new TypeError('entry (SearchEntry) required');
|
2011-09-16 16:06:35 +00:00
|
|
|
if (nofiltering === undefined)
|
|
|
|
nofiltering = false;
|
2012-02-18 08:15:52 +00:00
|
|
|
if (typeof (nofiltering) !== 'boolean')
|
2011-10-17 23:37:09 +00:00
|
|
|
throw new TypeError('noFiltering must be a boolean');
|
2011-08-04 20:32:01 +00:00
|
|
|
|
2011-08-12 23:37:47 +00:00
|
|
|
var self = this;
|
|
|
|
|
2011-09-27 18:49:33 +00:00
|
|
|
if (entry instanceof SearchEntry || entry instanceof SearchReference) {
|
|
|
|
if (!entry.messageID)
|
|
|
|
entry.messageID = this.messageID;
|
|
|
|
if (entry.messageID !== this.messageID)
|
|
|
|
throw new Error('SearchEntry messageID mismatch');
|
|
|
|
} else {
|
2011-08-12 23:37:47 +00:00
|
|
|
if (!entry.attributes)
|
|
|
|
throw new Error('entry.attributes required');
|
|
|
|
|
2011-11-22 21:29:19 +00:00
|
|
|
var savedAttrs = {};
|
2011-10-17 23:37:09 +00:00
|
|
|
var all = (self.attributes.indexOf('*') !== -1);
|
2012-02-18 08:15:52 +00:00
|
|
|
Object.keys(entry.attributes).forEach(function (a) {
|
2011-10-17 23:37:09 +00:00
|
|
|
var _a = a.toLowerCase();
|
|
|
|
if (!nofiltering && _a.length && _a[0] === '_') {
|
2011-11-22 21:29:19 +00:00
|
|
|
savedAttrs[a] = entry.attributes[a];
|
2011-10-17 23:37:09 +00:00
|
|
|
delete entry.attributes[a];
|
2011-10-18 19:31:52 +00:00
|
|
|
} else if (!nofiltering && self.notAttributes.indexOf(_a) !== -1) {
|
2011-11-22 21:29:19 +00:00
|
|
|
savedAttrs[a] = entry.attributes[a];
|
2011-10-17 23:37:09 +00:00
|
|
|
delete entry.attributes[a];
|
|
|
|
} else if (all) {
|
2012-02-18 08:15:52 +00:00
|
|
|
return;
|
2011-10-17 23:37:09 +00:00
|
|
|
} else if (self.attributes.length && self.attributes.indexOf(_a) === -1) {
|
2011-11-22 21:29:19 +00:00
|
|
|
savedAttrs[a] = entry.attributes[a];
|
2011-10-17 23:37:09 +00:00
|
|
|
delete entry.attributes[a];
|
|
|
|
}
|
|
|
|
});
|
2011-10-11 17:43:27 +00:00
|
|
|
|
2011-10-17 23:37:09 +00:00
|
|
|
var save = entry;
|
2011-08-12 23:37:47 +00:00
|
|
|
entry = new SearchEntry({
|
2012-02-18 08:15:52 +00:00
|
|
|
objectName: typeof (save.dn) === 'string' ? parseDN(save.dn) : save.dn,
|
2011-08-12 23:37:47 +00:00
|
|
|
messageID: self.messageID,
|
2012-02-18 08:54:22 +00:00
|
|
|
log: self.log
|
2011-08-12 23:37:47 +00:00
|
|
|
});
|
|
|
|
entry.fromObject(save);
|
|
|
|
}
|
2011-08-04 20:32:01 +00:00
|
|
|
|
2011-08-12 23:37:47 +00:00
|
|
|
try {
|
2012-02-18 08:54:22 +00:00
|
|
|
if (this.log.debug())
|
2011-08-12 23:37:47 +00:00
|
|
|
this.log.debug('%s: sending: %j', this.connection.ldap.id, entry.json);
|
|
|
|
|
|
|
|
this.connection.write(entry.toBer());
|
2011-09-13 21:49:21 +00:00
|
|
|
this.sentEntries++;
|
2011-11-15 18:49:23 +00:00
|
|
|
|
|
|
|
if (self._dtraceOp && self._dtraceId) {
|
2012-02-18 08:15:52 +00:00
|
|
|
dtrace.fire('server-search-entry', function () {
|
2011-11-15 18:49:23 +00:00
|
|
|
var c = self.connection || {ldap: {}};
|
|
|
|
return [
|
|
|
|
self._dtraceId || 0,
|
|
|
|
(c.remoteAddress || ''),
|
|
|
|
c.ldap.bindDN ? c.ldap.bindDN.toString() : '',
|
|
|
|
(self.requestDN ? self.requestDN.toString() : ''),
|
|
|
|
entry.objectName.toString(),
|
2011-11-22 21:29:19 +00:00
|
|
|
entry.attributes.length
|
2011-11-15 18:49:23 +00:00
|
|
|
];
|
|
|
|
});
|
|
|
|
}
|
2011-11-22 21:29:19 +00:00
|
|
|
|
|
|
|
// Restore attributes
|
2012-07-09 13:00:51 +00:00
|
|
|
Object.keys(savedAttrs || {}).forEach(function (k) {
|
2011-11-22 21:29:19 +00:00
|
|
|
save.attributes[k] = savedAttrs[k];
|
|
|
|
});
|
|
|
|
|
2011-08-12 23:37:47 +00:00
|
|
|
} catch (e) {
|
2012-07-09 13:00:51 +00:00
|
|
|
this.log.warn(e, '%s failure to write message %j',
|
|
|
|
this.connection.ldap.id, this.json);
|
2011-08-12 23:37:47 +00:00
|
|
|
}
|
2011-08-04 20:32:01 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2011-10-17 23:37:09 +00:00
|
|
|
|
2012-02-18 08:15:52 +00:00
|
|
|
SearchResponse.prototype.createSearchEntry = function (object) {
|
|
|
|
if (!object || typeof (object) !== 'object')
|
2011-08-12 23:37:47 +00:00
|
|
|
throw new TypeError('object required');
|
|
|
|
|
|
|
|
var self = this;
|
2011-08-04 20:32:01 +00:00
|
|
|
|
2011-08-12 23:37:47 +00:00
|
|
|
var entry = new SearchEntry({
|
|
|
|
messageID: self.messageID,
|
2012-02-18 08:54:22 +00:00
|
|
|
log: self.log,
|
2011-08-12 23:37:47 +00:00
|
|
|
objectName: object.objectName || object.dn
|
|
|
|
});
|
|
|
|
entry.fromObject((object.attributes || object));
|
2011-08-04 20:32:01 +00:00
|
|
|
|
2011-08-12 23:37:47 +00:00
|
|
|
return entry;
|
2011-08-04 20:32:01 +00:00
|
|
|
};
|
2011-09-27 18:49:33 +00:00
|
|
|
|
|
|
|
|
2012-02-18 08:15:52 +00:00
|
|
|
SearchResponse.prototype.createSearchReference = function (uris) {
|
2011-09-27 18:49:33 +00:00
|
|
|
if (!uris)
|
|
|
|
throw new TypeError('uris ([string]) required');
|
|
|
|
|
|
|
|
if (!Array.isArray(uris))
|
|
|
|
uris = [uris];
|
|
|
|
|
|
|
|
for (var i = 0; i < uris.length; i++) {
|
2012-02-18 08:15:52 +00:00
|
|
|
if (typeof (uris[i]) == 'string')
|
2011-09-27 18:49:33 +00:00
|
|
|
uris[i] = parseURL(uris[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
var self = this;
|
|
|
|
return new SearchReference({
|
|
|
|
messageID: self.messageID,
|
2012-02-18 08:54:22 +00:00
|
|
|
log: self.log,
|
2011-09-27 18:49:33 +00:00
|
|
|
uris: uris
|
|
|
|
});
|
|
|
|
};
|