#!/usr/bin/env node
// -*- mode: js -*-
// Copyright 2011 Mark Cavage.  All rights reserved.

var path = require('path');
var url = require('url');

var log4js = require('log4js');
var nopt = require('nopt');

var ldap = require('../lib/index');



///--- Globals

nopt.typeDefs.DN = {
  type: ldap.DN,
  validate: function(data, k, val) {
    data[k] = ldap.parseDN(val);
  }
};

nopt.typeDefs.Filter = {
  type: ldap.Filter,
  validate: function(data, k, val) {
    data[k] = ldap.parseFilter(val);
  }
};


var opts = {
  'debug': Number,
  'base': ldap.DN,
  'binddn': ldap.DN,
  'control': Array,
  'password': String,
  'scope': String,
  'url': url
};

var shortOpts = {
  'c': ['--control'],
  'd': ['--debug'],
  'b': ['--base'],
  'D': ['--binddn'],
  'w': ['--password'],
  's': ['--scope'],
  'u': ['--url']
};



///--- Helpers

function usage(code, message) {
  var _opts = '';
  Object.keys(shortOpts).forEach(function(k) {
    if (!Array.isArray(shortOpts[k]))
      return;
    var longOpt = shortOpts[k][0].replace('--', '');
    var type = opts[longOpt].name || 'string';
    if (type && type === 'boolean') type = '';
    type = type.toLowerCase();

    _opts += ' [--' + longOpt + ' ' + type + ']';
  });
  _opts += ' filter [attributes...]';

  var msg = (message ? message + '\n' : '') +
    'usage: ' + path.basename(process.argv[1]) + _opts;

  process.stderr.write(msg + '\n');
  process.exit(code);
}


function perror(err) {
  if (parsed.debug) {
    process.stderr.write(err.stack + '\n');
  } else {
    process.stderr.write(err.message + '\n');
  }
  process.exit(1);
}



///--- Mainline

log4js.setGlobalLogLevel('INFO');
var parsed;

try {
  parsed = nopt(opts, shortOpts, process.argv, 2);
} catch (e) {
  usage(1, e.toString());
}

if (parsed.help)
  usage(0);
if (parsed.argv.remain.length < 1)
  usage(1, 'filter required');

try {
  ldap.parseFilter(parsed.argv.remain[0]);
} catch (e) {
  usage(1, e.message);
}


if (parsed.debug)
  log4js.setGlobalLogLevel(parsed.debug > 1 ? 'TRACE' : 'DEBUG');
if (!parsed.url)
  parsed.url = 'ldap://127.0.0.1:389';
if (!parsed.binddn)
  parsed.binddn = '';
if (!parsed.password)
  parsed.password = '';
if (!parsed.base)
  parsed.base = '';
if (!parsed.control)
  parsed.control = [];

var client = ldap.createClient({
  url: parsed.url,
  log4js: log4js
});

client.on('error', function(err) {
  perror(err);
});

client.bind(parsed.binddn, parsed.password, function(err, res) {
  if (err)
    perror(err);

  var controls = [];
  parsed.control.forEach(function(c) {
    controls.push(new ldap.Control({
      type: c,
      criticality: true
    }));
  });
  var req = {
    scope: parsed.scope || 'sub',
    filter: parsed.argv.remain[0],
    attributes: parsed.argv.remain.length > 1 ? parsed.argv.remain.slice(1) : []
  };
  client.search(parsed.base, req, controls, function(err, res) {
    if (err)
      perror(err);

    var entries = [];
    res.on('searchEntry', function(entry) {
      entries.push(entry.object);
    });
    res.on('error', function(err) {
      perror(err);
    });
    res.on('end', function(res) {
      process.stdout.write(JSON.stringify(entries, null, 2) + '\n');
      if (res.status !== 0)
        process.stderr.write(ldap.getMessage(res.status) + '\n');
      client.unbind(function() {
        return;
      });
    });
  });
});