'use strict'

const { test } = require('tap')
const { dn } = require('../lib')

test('parse basic', function (t) {
  const DN_STR = 'cn=mark, ou=people, o=joyent'
  const name = dn.parse(DN_STR)
  t.ok(name)
  t.ok(name.rdns)
  t.ok(Array.isArray(name.rdns))
  t.equal(3, name.rdns.length)
  name.rdns.forEach(function (rdn) {
    t.equal('object', typeof (rdn))
  })
  t.equal(name.toString(), DN_STR)
  t.end()
})

test('parse escaped', function (t) {
  const DN_STR = 'cn=m\\,ark, ou=people, o=joyent'
  const name = dn.parse(DN_STR)
  t.ok(name)
  t.ok(name.rdns)
  t.ok(Array.isArray(name.rdns))
  t.equal(3, name.rdns.length)
  name.rdns.forEach(function (rdn) {
    t.equal('object', typeof (rdn))
  })
  t.equal(name.toString(), DN_STR)
  t.end()
})

test('parse compound', function (t) {
  const DN_STR = 'cn=mark+sn=cavage, ou=people, o=joyent'
  const name = dn.parse(DN_STR)
  t.ok(name)
  t.ok(name.rdns)
  t.ok(Array.isArray(name.rdns))
  t.equal(3, name.rdns.length)
  name.rdns.forEach(function (rdn) {
    t.equal('object', typeof (rdn))
  })
  t.equal(name.toString(), DN_STR)
  t.end()
})

test('parse quoted', function (t) {
  const DN_STR = 'cn="mark+sn=cavage", ou=people, o=joyent'
  const ESCAPE_STR = 'cn=mark\\+sn\\=cavage, ou=people, o=joyent'
  const name = dn.parse(DN_STR)
  t.ok(name)
  t.ok(name.rdns)
  t.ok(Array.isArray(name.rdns))
  t.equal(3, name.rdns.length)
  name.rdns.forEach(function (rdn) {
    t.equal('object', typeof (rdn))
  })
  t.equal(name.toString(), ESCAPE_STR)
  t.end()
})

test('equals', function (t) {
  const dn1 = dn.parse('cn=foo,dc=bar')
  t.ok(dn1.equals('cn=foo,dc=bar'))
  t.ok(!dn1.equals('cn=foo1,dc=bar'))
  t.ok(dn1.equals(dn.parse('cn=foo,dc=bar')))
  t.ok(!dn1.equals(dn.parse('cn=foo2,dc=bar')))
  t.end()
})

test('child of', function (t) {
  const dn1 = dn.parse('cn=foo,dc=bar')
  t.ok(dn1.childOf('dc=bar'))
  t.ok(!dn1.childOf('dc=moo'))
  t.ok(!dn1.childOf('dc=foo'))
  t.ok(!dn1.childOf('cn=foo,dc=bar'))

  t.ok(dn1.childOf(dn.parse('dc=bar')))
  t.end()
})

test('parent of', function (t) {
  const dn1 = dn.parse('cn=foo,dc=bar')
  t.ok(dn1.parentOf('cn=moo,cn=foo,dc=bar'))
  t.ok(!dn1.parentOf('cn=moo,cn=bar,dc=foo'))
  t.ok(!dn1.parentOf('cn=foo,dc=bar'))

  t.ok(dn1.parentOf(dn.parse('cn=moo,cn=foo,dc=bar')))
  t.end()
})

test('DN parent', function (t) {
  const _dn = dn.parse('cn=foo,ou=bar')
  const parent1 = _dn.parent()
  const parent2 = parent1.parent()
  t.ok(parent1.equals('ou=bar'))
  t.ok(parent2.equals(''))
  t.equal(parent2.parent(), null)
  t.end()
})

test('empty DNs', function (t) {
  const _dn = dn.parse('')
  const _dn2 = dn.parse('cn=foo')
  t.ok(_dn.isEmpty())
  t.notOk(_dn2.isEmpty())
  t.notOk(_dn.equals('cn=foo'))
  t.notOk(_dn2.equals(''))
  t.ok(_dn.parentOf('cn=foo'))
  t.notOk(_dn.childOf('cn=foo'))
  t.notOk(_dn2.parentOf(''))
  t.ok(_dn2.childOf(''))
  t.end()
})

test('case insensitive attribute names', function (t) {
  const dn1 = dn.parse('CN=foo,dc=bar')
  t.ok(dn1.equals('cn=foo,dc=bar'))
  t.ok(dn1.equals(dn.parse('cn=foo,DC=bar')))
  t.end()
})

test('format', function (t) {
  const DN_ORDER = dn.parse('sn=bar+cn=foo,ou=test')
  const DN_QUOTE = dn.parse('cn="foo",ou=test')
  const DN_QUOTE2 = dn.parse('cn=" foo",ou=test')
  const DN_SPACE = dn.parse('cn=foo,ou=test')
  const DN_SPACE2 = dn.parse('cn=foo ,ou=test')
  const DN_CASE = dn.parse('CN=foo,Ou=test')

  t.equal(DN_ORDER.format({ keepOrder: false }), 'cn=foo+sn=bar, ou=test')
  t.equal(DN_ORDER.format({ keepOrder: true }), 'sn=bar+cn=foo, ou=test')

  t.equal(DN_QUOTE.format({ keepQuote: false }), 'cn=foo, ou=test')
  t.equal(DN_QUOTE.format({ keepQuote: true }), 'cn="foo", ou=test')
  t.equal(DN_QUOTE2.format({ keepQuote: false }), 'cn=" foo", ou=test')
  t.equal(DN_QUOTE2.format({ keepQuote: true }), 'cn=" foo", ou=test')

  t.equal(DN_SPACE.format({ keepSpace: false }), 'cn=foo, ou=test')
  t.equal(DN_SPACE.format({ keepSpace: true }), 'cn=foo,ou=test')
  t.equal(DN_SPACE.format({ skipSpace: true }), 'cn=foo,ou=test')
  t.equal(DN_SPACE2.format({ keepSpace: false }), 'cn=foo, ou=test')
  t.equal(DN_SPACE2.format({ keepSpace: true }), 'cn=foo ,ou=test')
  t.equal(DN_SPACE2.format({ skipSpace: true }), 'cn=foo,ou=test')

  t.equal(DN_CASE.format({ keepCase: false }), 'cn=foo, ou=test')
  t.equal(DN_CASE.format({ keepCase: true }), 'CN=foo, Ou=test')
  t.equal(DN_CASE.format({ upperName: true }), 'CN=foo, OU=test')
  t.end()
})

test('set format', function (t) {
  const _dn = dn.parse('uid="user",  sn=bar+cn=foo, dc=test , DC=com')
  t.equal(_dn.toString(), 'uid=user, cn=foo+sn=bar, dc=test, dc=com')
  _dn.setFormat({ keepOrder: true })
  t.equal(_dn.toString(), 'uid=user, sn=bar+cn=foo, dc=test, dc=com')
  _dn.setFormat({ keepQuote: true })
  t.equal(_dn.toString(), 'uid="user", cn=foo+sn=bar, dc=test, dc=com')
  _dn.setFormat({ keepSpace: true })
  t.equal(_dn.toString(), 'uid=user,  cn=foo+sn=bar, dc=test , dc=com')
  _dn.setFormat({ keepCase: true })
  t.equal(_dn.toString(), 'uid=user, cn=foo+sn=bar, dc=test, DC=com')
  _dn.setFormat({ upperName: true })
  t.equal(_dn.toString(), 'UID=user, CN=foo+SN=bar, DC=test, DC=com')
  t.end()
})

test('format persists across clone', function (t) {
  const _dn = dn.parse('uid="user",  sn=bar+cn=foo, dc=test , DC=com')
  const OUT = 'UID="user", CN=foo+SN=bar, DC=test, DC=com'
  _dn.setFormat({ keepQuote: true, upperName: true })
  const clone = _dn.clone()
  t.equals(_dn.toString(), OUT)
  t.equals(clone.toString(), OUT)
  t.end()
})

test('initialization', function (t) {
  const dn1 = new dn.DN()
  t.ok(dn1)
  t.equals(dn1.toString(), '')
  t.ok(dn1.isEmpty(), 'DN with no initializer defaults to null DN')

  const data = [
    new dn.RDN({ foo: 'bar' }),
    new dn.RDN({ o: 'base' })
  ]
  const dn2 = new dn.DN(data)
  t.ok(dn2)
  t.equals(dn2.toString(), 'foo=bar, o=base')
  t.ok(!dn2.isEmpty())

  t.end()
})

test('array functions', function (t) {
  const dn1 = dn.parse('a=foo, b=bar, c=baz')
  t.ok(dn1)
  t.equal(dn1.toString(), 'a=foo, b=bar, c=baz')

  t.ok(dn1.reverse())
  t.equal(dn1.toString(), 'c=baz, b=bar, a=foo')

  let rdn = dn1.pop()
  t.ok(rdn)
  t.equal(dn1.toString(), 'c=baz, b=bar')

  t.ok(dn1.push(rdn))
  t.equal(dn1.toString(), 'c=baz, b=bar, a=foo')

  rdn = dn1.shift()
  t.ok(rdn)
  t.equal(dn1.toString(), 'b=bar, a=foo')

  t.ok(dn1.unshift(rdn))
  t.equal(dn1.toString(), 'c=baz, b=bar, a=foo')

  t.end()
})

test('isDN duck-testing', function (t) {
  const valid = dn.parse('cn=foo')
  const isDN = dn.DN.isDN
  t.notOk(isDN(null))
  t.notOk(isDN('cn=foo'))
  t.ok(isDN(valid))
  const duck = {
    rdns: [{ look: 'ma' }, { a: 'dn' }],
    toString: function () { return 'look=ma, a=dn' }
  }
  t.ok(isDN(duck))
  t.end()
})