(function() {
  'use strict';
  /*global window, document, location, URI, URI_pre_lib, IPv6, IPv6_pre_lib, URITemplate, URITemplate_pre_lib, SecondLevelDomains, SecondLevelDomains_pre_lib, urls, test, ok, equal, strictEqual, deepEqual, raises */
  // FIXME: v2.0.0 renamce non-camelCase properties to uppercase
  /*jshint camelcase: false, loopfunc: true */
  test('loaded', function() {
    ok(window.URI);
  });
  module('constructing');
  test('URI()', function() {
    var u = URI();
    ok(u instanceof URI, 'instanceof URI');
    equal(u.toString(), window.location && window.location.href || '', 'is location (browser) or empty string (node)');
  });
  test('URI(undefined)', function() {
    raises(function() {
      URI(undefined);
    }, TypeError, 'Failing undefined input');
  });
  test('new URI(string)', function() {
    var u = new URI('http://example.org/');
    ok(u instanceof URI, 'instanceof URI');
    ok(u._parts.hostname !== undefined, 'host undefined');
  });
  test('new URI(object)', function() {
    var u = new URI({protocol: 'http', hostname: 'example.org'});
    ok(u instanceof URI, 'instanceof URI');
    ok(u._parts.hostname !== undefined, 'host undefined');
  });
  test('new URI(Location)', function () {
    var u = new URI(location);
    equal(u.href(), String(location.href), 'location object');
  });
  test('new URI(undefined)', function() {
    var u = new URI();
    ok(u instanceof URI, 'instanceof URI');
    equal(u.toString(), window.location && window.location.href || '', 'is location (browser) or empty string (node)');
    raises(function() {
      new URI(undefined);
    }, TypeError, 'Failing undefined input');
  });
  (function() {
    var element;
    function testDomAttribute(element, attribute) {
      test('new URI(Element ' + element.nodeName + ')', function() {
        element[attribute] = 'http://example.org/foobar.html';
        var u = new URI(element);
        equal(u.scheme(), 'http', 'scheme');
        equal(u.host(), 'example.org', 'host');
        equal(u.path(), '/foobar.html', 'path');
        element[attribute] = 'file:///C:/foo/bar.html';
        u = new URI(element);
        equal(u.href(), element[attribute], 'file');
      });
    }
    function testUnsupportedDomAttribute(element, attribute) {
      test('new URI(unsupported Element ' + element.nodeName + ')', function() {
        element[attribute] = 'http://example.org/foobar.html';
        var u = new URI(element);
        equal(u.scheme(), '', 'scheme');
        equal(u.host(), '', 'host');
        equal(u.path(), '', 'path');
        element[attribute] = 'file:///C:/foo/bar.html';
        u = new URI(element);
        equal(u.href(), '', 'file');
      });
    }
    for (var nodeName in URI.domAttributes) {
      if (!Object.prototype.hasOwnProperty.call(URI.domAttributes, nodeName) || nodeName === 'input') {
        continue;
      }
      element = document.createElement(nodeName);
      testDomAttribute(element, URI.domAttributes[nodeName]);
    }
    element = document.createElement('input');
    element.type = 'image';
    testDomAttribute(element, 'src');
    element = document.createElement('input');
    testUnsupportedDomAttribute(element, 'src');
    element = document.createElement('div');
    testUnsupportedDomAttribute(element, 'src');
  })();
  test('new URI(URI)', function() {
    var u = new URI(new URI({protocol: 'http', hostname: 'example.org'}));
    ok(u instanceof URI, 'instanceof URI');
    ok(u._parts.hostname !== undefined, 'host undefined');
  });
  test('new URI(new Date())', function() {
    raises(function() {
      new URI(new Date());
    }, TypeError, 'Failing unknown input');
  });
  test('new URI(undefined)', function() {
    raises(function() {
      new URI(undefined);
    }, TypeError, 'Failing undefined input');
  });
  test('new URI()', function() {
    var u = new URI();
    ok(u instanceof URI, 'instanceof URI');
    ok(u._parts.hostname === location.hostname || u._parts.hostname === null && location.hostname === '',
        'hostname == location.hostname');
  });
  test('function URI(string)', function() {
    var u = new URI('http://example.org/');
    ok(u instanceof URI, 'instanceof URI');
    ok(u._parts.hostname !== undefined, 'host undefined');
  });
  test('new URI(string, string)', function() {
    // see http://dvcs.w3.org/hg/url/raw-file/tip/Overview.html#constructor
    var u = new URI('../foobar.html', 'http://example.org/hello/world.html');
    equal(u+'', 'http://example.org/foobar.html', 'resolve on construct');
  });
  module('parsing');
  // [].forEach() no IE, lacking interest in polyfilling this...
  for (var i = 0, t; (t = urls[i]); i++) {
    (function(t){
      test('parse ' + t.name, function() {
        var u = new URI(t.url),
          key;
        // test URL built from parts
        equal(u + '', t._url || t.url, 'toString');
        // test parsed parts
        for (key in t.parts) {
          if (Object.hasOwnProperty.call(t.parts, key)) {
            equal(u._parts[key], t.parts[key], 'part: ' + key);
          }
        }
        // test accessors
        for (key in t.accessors) {
          if (Object.hasOwnProperty.call(t.accessors, key)) {
            equal(u[key](), t.accessors[key], 'accessor: ' + key);
          }
        }
        // test is()
        for (key in t.is) {
          if (Object.hasOwnProperty.call(t.is, key)) {
            equal(u.is(key), t.is[key], 'is: ' + key);
          }
        }
      });
    })(t);
  }
  module('mutating basics');
  test('protocol', function() {
    var u = new URI('http://example.org/foo.html');
    u.protocol('ftp');
    equal(u.protocol(), 'ftp', 'ftp protocol');
    equal(u+'', 'ftp://example.org/foo.html', 'ftp url');
    u.protocol('');
    equal(u.protocol(), '', 'relative protocol');
    equal(u+'', '//example.org/foo.html', 'relative-scheme url');
    u.protocol('f.t-p+0');
    equal(u.protocol(), 'f.t-p+0', 'character profile');
    try {
      u.protocol('f:t');
      ok(false, 'do not accept invalid protocol');
    } catch(e) {}
    u.protocol(null);
    equal(u.protocol(), '', 'missing protocol');
    equal(u+'', '//example.org/foo.html', 'missing-scheme url');
  });
  test('username', function() {
    var u = new URI('http://example.org/foo.html');
    u.username('hello');
    equal(u.username(), 'hello', 'changed username hello');
    equal(u.password(), '', 'changed passowrd hello');
    equal(u+'', 'http://hello@example.org/foo.html', 'changed url hello');
    u.username('');
    equal(u.username(), '', 'changed username ""');
    equal(u.password(), '', 'changed passowrd ""');
    equal(u+'', 'http://example.org/foo.html', 'changed url ""');
  });
  test('password', function() {
    var u = new URI('http://hello@example.org/foo.html');
    u.password('world');
    equal(u.username(), 'hello', 'changed username world');
    equal(u.password(), 'world', 'changed passowrd world');
    equal(u+'', 'http://hello:world@example.org/foo.html', 'changed url world');
    u.password('');
    equal(u.username(), 'hello', 'changed username ""');
    equal(u.password(), '', 'changed passowrd ""');
    equal(u+'', 'http://hello@example.org/foo.html', 'changed url ""');
    u.username('').password('hahaha');
    equal(u.username(), '', 'changed username - password without username');
    equal(u.password(), 'hahaha', 'changed password - password without username');
    equal(u+'', 'http://:hahaha@example.org/foo.html', 'changed url - password without username');
  });
  test('hostname', function() {
    var u = new URI('http://example.org/foo.html');
    u.hostname('abc.foobar.lala');
    equal(u.hostname(), 'abc.foobar.lala', 'hostname changed');
    equal(u+'', 'http://abc.foobar.lala/foo.html', 'hostname changed url');
    u.hostname('');
    equal(u.hostname(), '', 'hostname removed');
    equal(u+'', 'http:///foo.html', 'hostname removed url');
    raises(function() {
      u.hostname('foo\\bar.com');
    }, TypeError, 'Failing backslash detection in hostname');
  });
  test('port', function() {
    var u = new URI('http://example.org/foo.html');
    u.port('80');
    equal(u.port(), '80', 'changing port 80');
    equal(u+'', 'http://example.org:80/foo.html', 'changing url 80');
    u.port('');
    equal(u.port(), '', 'changing port ""');
    equal(u+'', 'http://example.org/foo.html', 'changing url ""');
  });
  test('path', function() {
    var u = new URI('http://example.org/foobar.html?query=string');
    u.pathname('/some/path/file.suffix');
    equal(u.pathname(), '/some/path/file.suffix', 'changing pathname "/some/path/file.suffix"');
    equal(u+'', 'http://example.org/some/path/file.suffix?query=string', 'changing url "/some/path/file.suffix"');
    u.pathname('');
    equal(u.pathname(), '/', 'changing pathname ""');
    equal(u+'', 'http://example.org/?query=string', 'changing url ""');
    u.pathname('/~userhome/@mine;is %2F and/');
    equal(u.pathname(), '/~userhome/@mine;is%20%2F%20and/', 'path encoding');
    equal(u.pathname(true), '/~userhome/@mine;is %2F and/', 'path decoded');
    u = new URI('/a/b/c/').relativeTo('/a/b/c/');
    equal(u.pathname(), '', 'empty relative path');
    equal(u.toString(), '', 'empty relative path to string');
    u.pathname('/');
    equal(u.pathname(), '/', 'empty absolute path');
    equal(u.toString(), '/', 'empty absolute path to string');
  });
  test('URN paths', function() {
    var u = new URI('urn:uuid:6e8bc430-9c3a-11d9-9669-0800200c9a66?foo=bar');
    u.pathname('uuid:de305d54-75b4-431b-adb2-eb6b9e546013');
    equal(u.pathname(), 'uuid:de305d54-75b4-431b-adb2-eb6b9e546013');
    equal(u + '', 'urn:uuid:de305d54-75b4-431b-adb2-eb6b9e546013?foo=bar');
    u.pathname('');
    equal(u.pathname(), '', 'changing pathname ""');
    equal(u+'', 'urn:?foo=bar', 'changing url ""');
    u.pathname('music:classical:Béla Bártok%3a Concerto for Orchestra');
    equal(u.pathname(), 'music:classical:B%C3%A9la%20B%C3%A1rtok%3A%20Concerto%20for%20Orchestra', 'path encoding');
    equal(u.pathname(true), 'music:classical:Béla Bártok%3A Concerto for Orchestra', 'path decoded');
  });
  test('query', function() {
    var u = new URI('http://example.org/foo.html');
    u.query('foo=bar=foo');
    equal(u.query(), 'foo=bar=foo', 'query: foo=bar=foo');
    equal(u.search(), '?foo=bar=foo', 'query: foo=bar=foo - search');
    u.query('?bar=foo');
    equal(u.query(), 'bar=foo', 'query: ?bar=foo');
    equal(u.search(), '?bar=foo', 'query: ?bar=foo - search');
    u.query('');
    equal(u.query(), '', 'query: ""');
    equal(u.search(), '', 'query: "" - search');
    equal(u.toString(), 'http://example.org/foo.html');
    u.search('foo=bar=foo');
    equal(u.query(), 'foo=bar=foo', 'search: foo=bar=foo');
    equal(u.search(), '?foo=bar=foo', 'search: foo=bar=foo - query');
    u.search('?bar=foo');
    equal(u.query(), 'bar=foo', 'search: ?bar=foo');
    equal(u.search(), '?bar=foo', 'search: ?bar=foo - query');
    u.search('');
    equal(u.query(), '', 'search: ""');
    equal(u.search(), '', 'search: "" - query');
    u.query('?foo');
    equal(u.query(), 'foo', 'search: ""');
    equal(u.search(), '?foo', 'search: "" - query');
    u.search('foo=&foo=bar');
    equal(u.query(), 'foo=&foo=bar', 'search: foo=&foo=bar');
    equal(JSON.stringify(u.query(true)), JSON.stringify({foo: ['', 'bar']}), 'parsed query: {foo:["", "bar"]}');
    u.search('foo=bar&foo=');
    equal(u.query(), 'foo=bar&foo=', 'search: foo=bar&foo=');
    equal(JSON.stringify(u.query(true)), JSON.stringify({foo: ['bar', '']}), 'parsed query: {foo:["bar", ""]}');
    u.search('foo=bar&foo');
    equal(u.query(), 'foo=bar&foo', 'search: foo=bar&foo');
    equal(JSON.stringify(u.query(true)), JSON.stringify({foo: ['bar', null]}), 'parsed query: {foo:["bar", null]}');
    u.search('foo&foo=bar');
    equal(u.query(), 'foo&foo=bar', 'search: foo&foo=bar');
    equal(JSON.stringify(u.query(true)), JSON.stringify({foo: [null, 'bar']}), 'parsed query: {foo:[null, "bar"]}');
    // parsing empty query
    var t;
    t = u.query('?').query(true);
    t = u.query('').query(true);
    t = u.href('http://example.org').query(true);
  });
  test('fragment', function() {
    var u = new URI('http://example.org/foo.html');
    u.fragment('foo');
    equal(u.fragment(), 'foo', 'fragment: foo');
    equal(u.hash(), '#foo', 'fragment: foo - hash');
    u.fragment('#bar');
    equal(u.fragment(), 'bar', 'fragment: #bar');
    equal(u.hash(), '#bar', 'fragment: #bar - hash');
    u.fragment('');
    equal(u.fragment(), '', 'fragment: ""');
    equal(u.hash(), '', 'fragment: "" - hash');
    equal(u.toString(), 'http://example.org/foo.html');
    u.hash('foo');
    equal(u.fragment(), 'foo', 'hash: foo');
    equal(u.hash(), '#foo', 'hash: foo - fragment');
    u.hash('#bar');
    equal(u.fragment(), 'bar', 'hash: #bar');
    equal(u.hash(), '#bar', 'hash: #bar - fragment');
    u.hash('');
    equal(u.fragment(), '', 'hash: ""');
    equal(u.hash(), '', 'hash: "" - fragment');
  });
  module('mutating compounds');
  test('host', function() {
    var u = new URI('http://foo.bar/foo.html');
    u.host('example.org:80');
    equal(u.hostname(), 'example.org', 'host changed hostname');
    equal(u.port(), '80', 'host changed port');
    equal(u+'', 'http://example.org:80/foo.html', 'host changed url');
    u.host('some-domain.com');
    equal(u.hostname(), 'some-domain.com', 'host modified hostname');
    equal(u.port(), '', 'host removed port');
    equal(u+'', 'http://some-domain.com/foo.html', 'host modified url');
    raises(function() {
      u.host('foo\\bar.com');
    }, TypeError, 'Failing backslash detection in host');
  });
  test('origin', function () {
    var u = new URI('http://foo.bar/foo.html');
    equal(u.origin(), 'http://foo.bar', 'invalid origin');
    u.origin('http://bar.foo/bar.html');
    equal(u.origin(), 'http://bar.foo', 'origin didnt change');
    equal(u+'', 'http://bar.foo/foo.html', 'origin path changed');
  });
  test('authority', function() {
    var u = new URI('http://foo.bar/foo.html');
    u.authority('username:password@example.org:80');
    equal(u.username(), 'username', 'authority changed username');
    equal(u.password(), 'password', 'authority changed password');
    equal(u.hostname(), 'example.org', 'authority changed hostname');
    equal(u.port(), '80', 'authority changed port');
    equal(u+'', 'http://username:password@example.org:80/foo.html', 'authority changed url');
    u.authority('some-domain.com');
    equal(u.username(), '', 'authority removed username');
    equal(u.password(), '', 'authority removed password');
    equal(u.hostname(), 'some-domain.com', 'authority modified hostname');
    equal(u.port(), '', 'authority removed port');
    equal(u+'', 'http://some-domain.com/foo.html', 'authority modified url');
    raises(function() {
      u.authority('username:password@foo\\bar.com:80');
    }, TypeError, 'Failing backslash detection in authority');
  });
  test('userinfo', function() {
    var u = new URI('http://foo.bar/foo.html');
    u.userinfo('username:password');
    equal(u.username(), 'username', 'userinfo changed username-only');
    equal(u.password(), 'password', 'userinfo changed password');
    equal(u+'', 'http://username:password@foo.bar/foo.html', 'userinfo changed url');
    u.userinfo('walter');
    equal(u.username(), 'walter', 'userinfo removed password');
    equal(u.password(), '', 'userinfo removed password');
    equal(u+'', 'http://walter@foo.bar/foo.html', 'userinfo changed url');
    u.userinfo('');
    equal(u.username(), '', 'userinfo removed username');
    equal(u.password(), '', 'userinfo removed password');
    equal(u+'', 'http://foo.bar/foo.html', 'userinfo changed url');
  });
  test('href', function() {
    var u = new URI('http://foo.bar/foo.html');
    u.href('ftp://u:p@example.org:123/directory/file.suffix?query=string#fragment');
    equal(u.protocol(), 'ftp', 'href changed protocol');
    equal(u.username(), 'u', 'href changed username');
    equal(u.password(), 'p', 'href changed password');
    equal(u.hostname(), 'example.org', 'href changed hostname');
    equal(u.port(), '123', 'href changed port');
    equal(u.pathname(), '/directory/file.suffix', 'href changed pathname');
    equal(u.search(), '?query=string', 'href changed search');
    equal(u.hash(), '#fragment', 'href changed hash');
    equal(u.href(), 'ftp://u:p@example.org:123/directory/file.suffix?query=string#fragment', 'href removed url');
    u.href('../path/index.html');
    equal(u.protocol(), '', 'href removed protocol');
    equal(u.username(), '', 'href removed username');
    equal(u.password(), '', 'href removed password');
    equal(u.hostname(), '', 'href removed hostname');
    equal(u.port(), '', 'href removed port');
    equal(u.pathname(), '../path/index.html', 'href removed pathname');
    equal(u.search(), '', 'href removed search');
    equal(u.hash(), '', 'href removed hash');
    equal(u.href(), '../path/index.html', 'href removed url');
    /*jshint -W053 */
    u.href(new String('/narf'));
    /*jshint +W053 */
    equal(u.pathname(), '/narf', 'href from String instance');
  });
  test('resource', function() {
    var u = new URI('http://foo.bar/foo.html?hello#world');
    equal(u.resource(), '/foo.html?hello#world', 'get resource');
    u.resource('/foo.html?hello#world');
    equal(u.href(), 'http://foo.bar/foo.html?hello#world', 'set resource');
    u.resource('/world.html');
    equal(u.href(), 'http://foo.bar/world.html', 'set resource path');
    equal(u.resource(), '/world.html', 'get resource path');
    u.resource('?query');
    equal(u.href(), 'http://foo.bar/?query', 'set resource query');
    equal(u.resource(), '/?query', 'get resource query');
    u.resource('#fragment');
    equal(u.href(), 'http://foo.bar/#fragment', 'set resource fragment');
    equal(u.resource(), '/#fragment', 'get resource fragment');
    u.resource('?hello#world');
    equal(u.href(), 'http://foo.bar/?hello#world', 'set resource query+fragment');
    equal(u.resource(), '/?hello#world', 'get resource query+fragment');
    u.resource('/mars.txt?planet=123');
    equal(u.href(), 'http://foo.bar/mars.txt?planet=123', 'set resource path+query');
    equal(u.resource(), '/mars.txt?planet=123', 'get resource path+query');
    u.resource('/neptune.txt#foo');
    equal(u.href(), 'http://foo.bar/neptune.txt#foo', 'set resource path+fragment');
    equal(u.resource(), '/neptune.txt#foo', 'get resource path+fragment');
  });
  module('mutating fractions');
  test('subdomain', function() {
    var u = new URI('http://www.example.org/foo.html');
    u.subdomain('foo.bar');
    equal(u.hostname(), 'foo.bar.example.org', 'changed subdomain foo.bar');
    equal(u+'', 'http://foo.bar.example.org/foo.html', 'changed url foo.bar');
    u.subdomain('');
    equal(u.hostname(), 'example.org', 'changed subdomain ""');
    equal(u+'', 'http://example.org/foo.html', 'changed url ""');
    u.subdomain('foo.');
    equal(u.hostname(), 'foo.example.org', 'changed subdomain foo.');
    equal(u+'', 'http://foo.example.org/foo.html', 'changed url foo.');
  });
  test('domain', function() {
    var u = new URI('http://www.example.org/foo.html');
    u.domain('foo.bar');
    equal(u.hostname(), 'www.foo.bar', 'changed hostname foo.bar');
    equal(u+'', 'http://www.foo.bar/foo.html', 'changed url foo.bar');
    raises(function() {
      u.domain('');
    }, TypeError, 'Failing empty input');
    u.hostname('www.example.co.uk');
    equal(u.domain(), 'example.co.uk', 'domain after changed hostname www.example.co.uk');
    equal(u+'', 'http://www.example.co.uk/foo.html', 'url after changed hostname www.example.co.uk');
    equal(u.domain(true), 'co.uk', 'domain after changed hostname www.example.co.uk (TLD of SLD)');
    u.domain('example.org');
    equal(u.domain(), 'example.org', 'domain after changed domain example.org');
    equal(u+'', 'http://www.example.org/foo.html', 'url after changed domain example.org');
    u.domain('example.co.uk');
    equal(u.domain(), 'example.co.uk', 'domain after changed domain example.co.uk');
    equal(u+'', 'http://www.example.co.uk/foo.html', 'url after changed domain example.co.uk');
    u.href('http://test/');
    equal(u.domain(), 'test', 'domain (dot-less)');
    equal(u.subdomain(), '', 'subdomain (dot-less)');
    u.subdomain('foo');
    equal(u.href(), 'http://foo.test/', 'subdomain set on (dot-less)');
  });
  test('tld', function() {
    var u = new URI('http://www.example.org/foo.html');
    u.tld('mine');
    equal(u.tld(), 'mine', 'tld changed');
    equal(u+'', 'http://www.example.mine/foo.html', 'changed url mine');
    raises(function() {
      u.tld('');
    }, TypeError, 'Failing empty input');
    raises(function() {
      u.tld('foo.bar');
    }, TypeError, 'Failing "foo.bar"');
    u.tld('co.uk');
    equal(u.tld(), 'co.uk', 'tld changed to sld');
    equal(u+'', 'http://www.example.co.uk/foo.html', 'changed url to sld');
    equal(u.tld(true), 'uk', 'TLD of SLD');
    u.tld('org');
    equal(u.tld(), 'org', 'sld changed to tld');
    equal(u+'', 'http://www.example.org/foo.html', 'changed url to tld');
    u.hostname('www.examplet.se');
    equal(u.tld(), 'se', 'se tld');
  });
  test('sld', function() {
    // Lets just test them all..
    // Calling URI.is(), URI.domain(), URI.subdomain() allows us to indirectly
    // test SLD.has(), SLD.is() and SLD.get()
    var u = new URI('http://www.example.org/foo.html');
    equal(u.is('sld'), false, 'is not sld');
    var list = SecondLevelDomains.list;
    var tlds = Object.keys(list);
    var iTld = tlds.length;
    var tld, slds, sld, iSld;
    while ( iTld-- ) {
      tld = tlds[iTld];
      slds = list[tld].trim().split(/\s+/);
      iSld = slds.length;
      while ( iSld-- ) {
        sld = slds[iSld].trim() + '.' + tld;
        u.hostname('www.example.' + sld);
        equal(u.is('sld'), true, 'is sld');
        equal(u.domain(), 'example.' + sld, 'domain is example.' + sld);
        equal(u.subdomain(), 'www', 'subdomain is www');
        u.hostname('www.example.' + tld);
        equal(u.is('sld'), false, 'is not sld');
      }
    }
  });
  test('directory', function() {
    var u = new URI('http://www.example.org/some/directory/foo.html');
    u.directory('/');
    equal(u.path(), '/foo.html', 'changed path '/'');
    equal(u+'', 'http://www.example.org/foo.html', 'changed url '/'');
    u.directory('');
    equal(u.path(), '/foo.html', 'changed path ""');
    equal(u+'', 'http://www.example.org/foo.html', 'changed url ""');
    u.directory('/bar');
    equal(u.path(), '/bar/foo.html', 'changed path "/bar"');
    equal(u+'', 'http://www.example.org/bar/foo.html', 'changed url "/bar"');
    u.directory('baz');
    equal(u.path(), '/baz/foo.html', 'changed path "baz"');
    equal(u+'', 'http://www.example.org/baz/foo.html', 'changed url "baz"');
    // relative paths
    u = new URI('../some/directory/foo.html');
    u.directory('../other/');
    equal(u.path(), '../other/foo.html', 'changed path "../other/"');
    equal(u+'', '../other/foo.html', 'changed url "../other/"');
    u.directory('mine');
    equal(u.path(), 'mine/foo.html', 'changed path "mine"');
    equal(u+'', 'mine/foo.html', 'changed url "mine"');
    u.directory('/');
    equal(u.path(), '/foo.html', 'changed path "/"');
    equal(u+'', '/foo.html', 'changed url "/"');
    u.directory('');
    equal(u.path(), 'foo.html', 'changed path ""');
    equal(u+'', 'foo.html', 'changed url ""');
    u.directory('../blubb');
    equal(u.path(), '../blubb/foo.html', 'changed path "../blubb"');
    equal(u+'', '../blubb/foo.html', 'changed url "../blubb"');
    // encoding
    u.path('/some/directory/foo.html');
    u.directory('/~userhome/@mine;is %2F and/');
    equal(u.path(), '/~userhome/@mine;is%20%2F%20and/foo.html', 'directory encoding');
    equal(u.directory(true), '/~userhome/@mine;is %2F and', 'directory decoded');
  });
  test('filename', function() {
    var u = new URI('http://www.example.org/some/directory/foo.html');
    u.filename('hello.world');
    equal(u.path(), '/some/directory/hello.world', 'changed path "hello.world"');
    equal(u+'', 'http://www.example.org/some/directory/hello.world', 'changed url "hello.world"');
    u.filename('hello');
    equal(u.path(), '/some/directory/hello', 'changed path "hello"');
    equal(u+'', 'http://www.example.org/some/directory/hello', 'changed url "hello"');
    u.filename('');
    equal(u.path(), '/some/directory/', 'changed path ""');
    equal(u+'', 'http://www.example.org/some/directory/', 'changed url ""');
    u.filename('world');
    equal(u.path(), '/some/directory/world', 'changed path "world"');
    equal(u+'', 'http://www.example.org/some/directory/world', 'changed url "world"');
    // encoding
    u.path('/some/directory/foo.html');
    u.filename('hällo wörld.html');
    equal(u.path(), '/some/directory/h%C3%A4llo%20w%C3%B6rld.html', 'filename encoding');
    equal(u.filename(true), 'hällo wörld.html', 'filename decoded');
  });
  test('suffix', function() {
    var u = new URI('http://www.example.org/some/directory/foo.html');
    u.suffix('xml');
    equal(u.path(), '/some/directory/foo.xml', 'changed path "xml"');
    equal(u+'', 'http://www.example.org/some/directory/foo.xml', 'changed url "xml"');
    u.suffix('');
    equal(u.path(), '/some/directory/foo', 'changed path ""');
    equal(u+'', 'http://www.example.org/some/directory/foo', 'changed url ""');
    u.suffix('html');
    equal(u.path(), '/some/directory/foo.html', 'changed path "html"');
    equal(u+'', 'http://www.example.org/some/directory/foo.html', 'changed url "html"');
    // encoding
    u.suffix('cört');
    equal(u.path(), '/some/directory/foo.c%C3%B6rt', 'suffix encoding');
    equal(u.suffix(), 'c%C3%B6rt', 'suffix encoded'); // suffix is expected to be alnum!
    equal(u.suffix(true), 'cört', 'suffix decoded'); // suffix is expected to be alnum!
  });
  test('segment', function() {
    var u = new URI('http://www.example.org/some/directory/foo.html'),
      s = u.segment();
    equal(s.join('||'), 'some||directory||foo.html', 'segment get array');
    u.segment(['hello', 'world', 'foo.html']);
    equal(u.path(), '/hello/world/foo.html', 'segment set array');
    equal(u.segment(0), 'hello', 'segment get 0');
    equal(u.segment(2), 'foo.html', 'segment get 2');
    equal(u.segment(3), undefined, 'segment get 3');
    u.segment(0, 'goodbye');
    equal(u.path(), '/goodbye/world/foo.html', 'segment set 0');
    u.segment(2, 'bar.html');
    equal(u.path(), '/goodbye/world/bar.html', 'segment set 2');
    u.segment(3, 'zupp');
    equal(u.path(), '/goodbye/world/bar.html/zupp', 'segment set 3');
    u.segment('zapp');
    equal(u.path(), '/goodbye/world/bar.html/zupp/zapp', 'segment append');
    u.segment(3, '');
    equal(u.path(), '/goodbye/world/bar.html/zapp', 'segment del 3 ""');
    u.segment(3, null);
    equal(u.path(), '/goodbye/world/bar.html', 'segment del 3 null');
    u = new URI('http://www.example.org/some/directory/foo.html');
    equal(u.segment(-1), 'foo.html', 'segment get -1');
    u.segment(-1, 'world.html');
    equal(u.path(), '/some/directory/world.html', 'segment set -1');
    u = new URI('someurn:foo:bar:baz');
    equal(u.segment().join('||'), 'foo||bar||baz', 'segment get array URN');
    u.segment(1, 'mars');
    equal(u.path(), 'foo:mars:baz', 'segment set 1 URN');
    equal(u.toString(), 'someurn:foo:mars:baz', 'segment set 1 URN');
    u = new URI('/foo/');
    equal(u.segment().join('||'), 'foo||', 'segment get array trailing empty');
    u.segment('test');
    equal(u.path(), '/foo/test', 'segment append trailing empty');
    u.segment('');
    equal(u.path(), '/foo/test/', 'segment append empty trailing');
    u.segment('');
    equal(u.path(), '/foo/test/', 'segment append empty trailing unchanged');
    u.segment(['', '', 'foo', '', '', 'bar', '', '']);
    equal(u.path(), '/foo/bar/', 'segment collapsing empty parts');
    u = new URI('https://google.com');
    u.segment('//font.ttf//');
    equal(u.path(), '/font.ttf', 'segment removes trailing and leading slashes');
    u.segment(['/hello', '/world/', '//foo.html']);
    equal(u.path(), '/hello/world/foo.html', 'segment set array trimming slashes');
    u.segment(1, '/mars/');
    equal(u.path(), '/hello/mars/foo.html', 'segment set index trimming slashes');
  });
  test('segmentCoded', function() {
    var u = new URI('http://www.example.org/some%20thing/directory/foo.html'),
      s = u.segmentCoded();
    equal(s.join('||'), 'some thing||directory||foo.html', 'segmentCoded get array');
    u.segmentCoded(['hello/world']);
    equal(u.path(), '/hello%2Fworld', 'escape in array');
    u.segmentCoded('hello/world');
    equal(u.path(), '/hello%2Fworld/hello%2Fworld', 'escape appended value');
    u.segmentCoded(['hello world', 'mars', 'foo.html']);
    equal(u.path(), '/hello%20world/mars/foo.html', 'segmentCoded set array');
    equal(u.segmentCoded(0), 'hello world', 'segmentCoded get 0');
    equal(u.segmentCoded(2), 'foo.html', 'segmentCoded get 2');
    equal(u.segmentCoded(3), undefined, 'segmentCoded get 3');
    u.segmentCoded('zapp zerapp');
    equal(u.path(), '/hello%20world/mars/foo.html/zapp%20zerapp', 'segmentCoded append');
    u.segmentCoded(2, '');
    equal(u.path(), '/hello%20world/mars/zapp%20zerapp', 'segmentCoded del 3 ""');
    u.segmentCoded(2, null);
    equal(u.path(), '/hello%20world/mars', 'segmentCoded del 3 null');
    u.segmentCoded('');
    equal(u.path(), '/hello%20world/mars/', 'segmentCoded append empty trailing');
    u.segmentCoded('');
    equal(u.path(), '/hello%20world/mars/', 'segmentCoded append empty trailing unchanged');
  });
  module('mutating query strings');
  test('mutating object', function() {
    var u = new URI('?foo=bar&baz=bam&baz=bau'),
      q = u.query(true);
    q.something = ['new', 'and', 'funky'];
    u.query(q);
    equal(u.query(), 'foo=bar&baz=bam&baz=bau&something=new&something=and&something=funky', 'adding array');
    q.foo = undefined;
    u.query(q);
    equal(u.query(), 'baz=bam&baz=bau&something=new&something=and&something=funky', 'removing field');
    q.baz = undefined;
    u.query(q);
    equal(u.query(), 'something=new&something=and&something=funky', 'removing array');
  });
  test('query callback', function() {
    var u = URI('?foo=bar');
    u.query(function(data) {
      data.foo = 'bam';
    });
    equal(u.query(), 'foo=bam', 'augment argument');
    u.query(function() {
      return {
        bla: 'blubb'
      };
    });
    equal(u.query(), 'bla=blubb', 'overwrite returned value');
  });
  test('setQuery', function() {
    var u = URI('?foo=bar');
    u.setQuery('foo', 'bam');
    equal(u.query(), 'foo=bam', 'set name, value');
    u.setQuery('array', ['one', 'two']);
    equal(u.query(), 'foo=bam&array=one&array=two', 'set name, array');
    u.query('?foo=bar');
    u.setQuery({'obj': 'bam', foo: 'baz'});
    equal(u.query(), 'foo=baz&obj=bam', 'set {name: value}');
    u.setQuery({'foo': 'foo', bar: ['1', '2']});
    equal(u.query(), 'foo=foo&obj=bam&bar=1&bar=2', 'set {name: array}');
    u.query('?foo=bar');
    u.setQuery({'bam': null, 'baz': ''});
    equal(u.query(), 'foo=bar&bam&baz=', 'set {name: null}');
    u.query('?foo=bar');
    u.setQuery('empty');
    equal(u.query(), 'foo=bar&empty', 'set undefined');
    u.query('?foo=bar');
    u.setQuery('empty', '');
    equal(u.query(), 'foo=bar&empty=', 'set empty string');
    u.query('');
    u.setQuery('some value', 'must be encoded because of = and ? and #');
    equal(u.query(), 'some+value=must+be+encoded+because+of+%3D+and+%3F+and+%23', 'encoding');
    equal(u.query(true)['some value'], 'must be encoded because of = and ? and #', 'decoding');
  });
  test('addQuery', function() {
    var u = URI('?foo=bar');
    u.addQuery('baz', 'bam');
    equal(u.query(), 'foo=bar&baz=bam', 'add name, value');
    u.addQuery('array', ['one', 'two']);
    equal(u.query(), 'foo=bar&baz=bam&array=one&array=two', 'add name, array');
    u.query('?foo=bar');
    u.addQuery({'obj': 'bam', foo: 'baz'});
    equal(u.query(), 'foo=bar&foo=baz&obj=bam', 'add {name: value}');
    u.addQuery({'foo': 'bam', bar: ['1', '2']});
    equal(u.query(), 'foo=bar&foo=baz&foo=bam&obj=bam&bar=1&bar=2', 'add {name: array}');
    u.query('?foo=bar');
    u.addQuery({'bam': null, 'baz': ''});
    equal(u.query(), 'foo=bar&bam&baz=', 'add {name: null}');
    u.query('?foo=bar');
    u.addQuery('empty');
    equal(u.query(), 'foo=bar&empty', 'add undefined');
    u.query('?foo=bar');
    u.addQuery('empty', '');
    equal(u.query(), 'foo=bar&empty=', 'add empty string');
    u.query('?foo');
    u.addQuery('foo', 'bar');
    equal(u.query(), 'foo=bar', 'add to null value');
    u.query('');
    u.addQuery('some value', 'must be encoded because of = and ? and #');
    equal(u.query(), 'some+value=must+be+encoded+because+of+%3D+and+%3F+and+%23', 'encoding');
    equal(u.query(true)['some value'], 'must be encoded because of = and ? and #', 'decoding');
  });
  test('removeQuery', function() {
    var u = new URI('?foo=bar&foo=baz&foo=bam&obj=bam&bar=1&bar=2&bar=3');
    u.removeQuery('foo', 'bar');
    equal(u.query(), 'foo=baz&foo=bam&obj=bam&bar=1&bar=2&bar=3', 'removing name, value');
    u.removeQuery('foo');
    equal(u.query(), 'obj=bam&bar=1&bar=2&bar=3', 'removing name');
    u.removeQuery('bar', ['1', '3']);
    equal(u.query(), 'obj=bam&bar=2', 'removing name, array');
    u.query('?obj=bam&bar=1&bar=2');
    u.removeQuery('bar', ['2']);
    equal(u.query(), 'obj=bam&bar=1', 'removing name, singleton array');
    u.removeQuery('bar', ['1']);
    equal(u.query(), 'obj=bam', 'removing the last value via name, singleton array');
    u.query('?foo=one&foo=two').removeQuery('foo', ['one', 'two']);
    equal(u.query(), '', 'removing name, array, finishes empty');
    u.query('?foo=one,two').removeQuery('foo', ['one', 'two']);
    equal(u.query(), 'foo=one%2Ctwo', 'not removing name, array');
    u.query('?foo=one,two').removeQuery('foo', ['one,two']);
    equal(u.query(), '', 'removing name, singleton array with comma in value');
    u.query('?foo=bar&foo=baz&foo=bam&obj=bam&bar=1&bar=2&bar=3');
    u.removeQuery(['foo', 'bar']);
    equal(u.query(), 'obj=bam', 'removing array');
    u.query('?bar=1&bar=2');
    u.removeQuery({ bar: 1 });
    equal(u.query(), 'bar=2', 'removing non-string value from array');
    u.removeQuery({ bar: 2 });
    equal(u.query(), '', 'removing a non-string value');
    u.query('?foo=bar&foo=baz&foo=bam&obj=bam&bar=1&bar=2&bar=3');
    u.removeQuery({foo: 'bar', obj: undefined, bar: ['1', '2']});
    equal(u.query(), 'foo=baz&foo=bam&bar=3', 'removing object');
    u.query('?foo=bar&foo=baz&foo=bam&obj=bam&bar=1&bar=2&bar=3');
    u.removeQuery(/^bar/);
    equal(u.query(), 'foo=bar&foo=baz&foo=bam&obj=bam', 'removing by RegExp');
    u.query('?foo=bar&foo=baz&foo=bam&obj=bam&bar=bar&bar=baz&bar=bam');
    u.removeQuery('foo', /[rz]$/);
    equal(u.query(), 'foo=bam&obj=bam&bar=bar&bar=baz&bar=bam', 'removing by value RegExp');
  });
  test('duplicateQueryParameters', function() {
    var u = new URI('?bar=1&bar=1&bar=1');
    u.normalizeQuery();
    equal(u.toString(), '?bar=1', 'parameters de-duplicated');
    u = new URI('?bar=1&bar=1&bar=1');
    u.duplicateQueryParameters(true);
    ok(u._parts.duplicateQueryParameters, 'duplicateQueryParameters enabled');
    u.normalizeQuery();
    equal(u.toString(), '?bar=1&bar=1&bar=1', 'parameters NOT de-duplicated');
    ok(u._parts.duplicateQueryParameters, 'duplicateQueryParameters still enabled after normalizeQuery()');
    u.duplicateQueryParameters(false);
    u.normalizeQuery();
    equal(u.toString(), '?bar=1', 'parameters de-duplicated again');
    ok(!u._parts.duplicateQueryParameters, 'duplicateQueryParameters still disabled after normalizeQuery()');
    URI.duplicateQueryParameters = true;
    u = new URI('?bar=1&bar=1&bar=1');
    u.normalizeQuery();
    equal(u.toString(), '?bar=1&bar=1&bar=1', 'global configuration');
    URI.duplicateQueryParameters = false;
    // test cloning
    u = new URI('?bar=1&bar=1&bar=1');
    u = u.duplicateQueryParameters(true).clone();
    ok(u._parts.duplicateQueryParameters, 'duplicateQueryParameters still enabled after clone()');
    u.normalizeQuery();
    equal(u.toString(), '?bar=1&bar=1&bar=1', 'parameters NOT de-duplicated');
    // test adding
    u = new URI('?bar=1&bar=1&bar=1');
    u.duplicateQueryParameters(true);
    u.addQuery('bar', 1);
    equal(u.toString(), '?bar=1&bar=1&bar=1&bar=1', 'parameters NOT de-duplicated after addQuery()');
  });
  test('escapeQuerySpace', function() {
    var u = new URI('?bar=foo+bar&bam+baz=foo');
    var data = u.query(true);
    equal(data.bar, 'foo bar', 'value un-spac-escaped');
    equal(data['bam baz'], 'foo', 'name un-spac-escaped');
    u.escapeQuerySpace(false);
    data = u.query(true);
    equal(data.bar, 'foo+bar', 'value not un-spac-escaped');
    equal(data['bam+baz'], 'foo', 'name not un-spac-escaped');
    u.escapeQuerySpace(true);
    data = u.query(true);
    equal(data.bar, 'foo bar', 'value un-spac-escaped again');
    equal(data['bam baz'], 'foo', 'name un-spac-escaped again');
    u.escapeQuerySpace(false);
    u.addQuery('alpha bravo', 'charlie delta');
    equal(u.toString(), '?bar=foo%2Bbar&bam%2Bbaz=foo&alpha%20bravo=charlie%20delta', 'serialized un/escaped space');
    URI.escapeQuerySpace = false;
    u = new URI('?bar=foo+bar&bam+baz=foo');
    data = u.query(true);
    equal(data.bar, 'foo+bar', 'value not un-spac-escaped by default');
    equal(data['bam+baz'], 'foo', 'name not un-spac-escaped by default');
    // reset
    URI.escapeQuerySpace = true;
  });
  test('hasQuery', function() {
    var u = URI('?string=bar&list=one&list=two&number=123&null&empty=&nested[one]=1&nested[two]=2');
    // exists
    equal(u.hasQuery('string'), true, 'simple exists check - passing');
    equal(u.hasQuery('nono'), false, 'simple exists check - failing');
    // truthy value
    equal(u.hasQuery('string', true), true, 'has truthy value check - passing string');
    equal(u.hasQuery('number', true), true, 'has truthy value check - passing number');
    equal(u.hasQuery('list', true), true, 'has truthy value check - passing list');
    equal(u.hasQuery('empty', true), false, 'has truthy value check - failing empty');
    equal(u.hasQuery('null', true), false, 'has truthy value check - failing null');
    // falsy value
    equal(u.hasQuery('string', false), false, 'has falsy value check - failing string');
    equal(u.hasQuery('number', false), false, 'has falsy value check - failing number');
    equal(u.hasQuery('list', false), false, 'has falsy value check - failing list');
    equal(u.hasQuery('empty', false), true, 'has falsy value check - passing empty');
    equal(u.hasQuery('null', false), true, 'has falsy value check - passing null');
    // match value
    equal(u.hasQuery('string', 'bar'), true, 'value check - passing string');
    equal(u.hasQuery('number', 123), true, 'value check - passing number');
    equal(u.hasQuery('number', '123'), true, 'value check - passing number as string');
    equal(u.hasQuery('list', 'one'), false, 'value check - failing list');
    equal(u.hasQuery('empty', ''), true, 'value check - passing empty');
    equal(u.hasQuery('null', ''), false, 'value check - failing null');
    // matching RegExp
    equal(u.hasQuery('string', /ar$/), true, 'RegExp check - passing string');
    equal(u.hasQuery('number', /2/), true, 'RegExp check - passing number');
    equal(u.hasQuery('string', /nono/), false, 'RegExp check - failing string');
    equal(u.hasQuery('number', /999/), false, 'RegExp check - failing number');
    equal(u.hasQuery(/^nested/), true, 'RegExp name check - passing');
    equal(u.hasQuery(/^nested/, 2), true, 'RegExp name and value - passing number');
    equal(u.hasQuery(/^nested/, '2'), true, 'RegExp name and value - passing number as string');
    equal(u.hasQuery(/^nested/, 'nono'), false, 'RegExp name and value - failing string');
    equal(u.hasQuery(/^nested/, /2/), true, 'RegExp name and value - passing RegExp number');
    equal(u.hasQuery(/^nested/, /3/), false, 'RegExp name and value exists check - failing');
    equal(u.hasQuery(/^lis/, ['one']), false, 'RegExp name andarray check - failing incomplete list');
    equal(u.hasQuery(/^lis/, ['one', 'two']), true, 'RegExp name and array check - passing list');
    // matching array
    equal(u.hasQuery('string', ['one']), false, 'array check - failing string');
    equal(u.hasQuery('list', ['one']), false, 'array check - failing incomplete list');
    equal(u.hasQuery('list', ['one', 'two']), true, 'array check - passing list');
    equal(u.hasQuery('list', ['two', 'one']), true, 'array check - passing unsorted list');
    // matching part of array
    equal(u.hasQuery('string', ['one'], true), false, 'in array check - failing string');
    equal(u.hasQuery('list', 'one', true), true, 'in array check - passing value');
    equal(u.hasQuery('list', ['one'], true), true, 'in array check - passing incomplete list');
    equal(u.hasQuery('list', ['one', 'two'], true), true, 'in array check - passing list');
    equal(u.hasQuery('list', ['two', 'one'], true), true, 'in array check - passing unsorted list');
    equal(u.hasQuery('list', /ne$/, true), true, 'in array check - passing RegExp');
    equal(u.hasQuery('list', [/ne$/], true), true, 'in array check - passing RegExp list');
    // comparison function
    equal(u.hasQuery('string', function(value, name, data) {
      equal(value, 'bar', 'Function check - param value');
      equal(name, 'string', 'Function check - param name');
      equal(typeof data, 'object', 'Function check - param data');
      return true;
    }), true, 'Function check - passing true');
    equal(u.hasQuery('string', function() {
      return false;
    }), false, 'Function check - passing false');
  });
  module('normalizing');
  test('normalize', function() {
    var u = new URI('http://www.exämple.org:80/food/woo/.././../baz.html?&foo=bar&&baz=bam&&baz=bau');
    u.normalize();
    equal(u+'', 'http://www.xn--exmple-cua.org/baz.html?foo=bar&baz=bam&baz=bau', 'fully normalized URL');
  });
  test('normalizeProtocol', function() {
    var u = new URI('hTTp://example.org/foobar.html');
    u.normalizeProtocol();
    equal(u+'', 'http://example.org/foobar.html', 'lowercase http');
  });
  test('normalizeHost', function() {
    var u;
    if (window.punycode) {
      u = new URI('http://exämple.org/foobar.html');
      u.normalizeHostname();
      equal(u+'', 'http://xn--exmple-cua.org/foobar.html', 'converting IDN to punycode');
    }
    if (window.IPv6) {
      u = new URI('http://[fe80:0000:0000:0000:0204:61ff:fe9d:f156]/foobar.html');
      u.normalizeHostname();
      equal(u+'', 'http://[fe80::204:61ff:fe9d:f156]/foobar.html', 'best IPv6 representation');
    }
    if (window.IPv6) {
      u = new URI('http://[::1]/foobar.html');
      u.normalizeHostname();
      equal(u+'', 'http://[::1]/foobar.html', 'best IPv6 representation');
    }
    u = new URI('http://wWw.eXamplE.Org/foobar.html');
    u.normalizeHostname();
    equal(u+'', 'http://www.example.org/foobar.html', 'lower case hostname');
  });
  test('normalizePort', function() {
    var u = new URI('http://example.org:80/foobar.html');
    u.normalizePort();
    equal(u+'', 'http://example.org/foobar.html', 'dropping port 80 for http');
    u = new URI('ftp://example.org:80/foobar.html');
    u.normalizePort();
    equal(u+'', 'ftp://example.org:80/foobar.html', 'keeping port 80 for ftp');
  });
  test('normalizePath', function() {
    // relative URL
    var u = new URI('/food/bar/baz.html');
    u.normalizePath();
    equal(u.path(), '/food/bar/baz.html', 'absolute path without change');
    u.path('food/bar/baz.html').normalizePath();
    equal(u.path(), 'food/bar/baz.html', 'relative path without change');
    u.path('/food/../bar/baz.html').normalizePath();
    equal(u.path(), '/bar/baz.html', 'single parent');
    u.path('/food/woo/../../bar/baz.html').normalizePath();
    equal(u.path(), '/bar/baz.html', 'double parent');
    u.path('/food/woo/../bar/../baz.html').normalizePath();
    equal(u.path(), '/food/baz.html', 'split double parent');
    u.path('/food/woo/.././../baz.html').normalizePath();
    equal(u.path(), '/baz.html', 'cwd-split double parent');
    u.path('food/woo/../bar/baz.html').normalizePath();
    equal(u.path(), 'food/bar/baz.html', 'relative parent');
    u.path('./food/woo/../bar/baz.html').normalizePath();
    equal(u.path(), 'food/bar/baz.html', 'dot-relative parent');
    // absolute URL
    u = new URI('http://example.org/foo/bar/baz.html');
    u.normalizePath();
    equal(u.path(), '/foo/bar/baz.html', 'URL: absolute path without change');
    u.path('foo/bar/baz.html').normalizePath();
    equal(u.path(), '/foo/bar/baz.html', 'URL: relative path without change');
    u.path('/foo/../bar/baz.html').normalizePath();
    equal(u.path(), '/bar/baz.html', 'URL: single parent');
    u.path('/foo/woo/../../bar/baz.html').normalizePath();
    equal(u.path(), '/bar/baz.html', 'URL: double parent');
    u.path('/foo/woo/../bar/../baz.html').normalizePath();
    equal(u.path(), '/foo/baz.html', 'URL: split double parent');
    u.path('/foo/woo/.././../baz.html').normalizePath();
    equal(u.path(), '/baz.html', 'URL: cwd-split double parent');
    u.path('foo/woo/../bar/baz.html').normalizePath();
    equal(u.path(), '/foo/bar/baz.html', 'URL: relative parent');
    u.path('./foo/woo/../bar/baz.html').normalizePath();
    equal(u.path(), '/foo/bar/baz.html', 'URL: dot-relative parent');
    u.path('/.//').normalizePath();
    equal(u.path(), '/', 'root /.//');
    // encoding
    u._parts.path = '/~userhome/@mine;is %2F and/';
    u.normalize();
    equal(u.pathname(), '/~userhome/@mine;is%20%2F%20and/', 'path encoding');
    // relative URL
    u = URI('/.').normalizePath();
    equal(u.path(), '/', 'root /.');
    u = URI('/..').normalizePath();
    equal(u.path(), '/', 'root /..');
    u = URI('/foo/.').normalizePath();
    equal(u.path(), '/foo/', 'root /foo/.');
    u = URI('/foo/..').normalizePath();
    equal(u.path(), '/', 'root /foo/..');
    u = URI('/foo/.bar').normalizePath();
    equal(u.path(), '/foo/.bar', 'root /foo/.bar');
    u = URI('/foo/..bar').normalizePath();
    equal(u.path(), '/foo/..bar', 'root /foo/..bar');
    // Percent Encoding normalization has to happen before dot segment normalization
    u = URI('/foo/%2E%2E').normalizePath();
    equal(u.path(), '/', 'root /foo/%2E%2E');
    u = URI('/foo/%2E').normalizePath();
    equal(u.path(), '/foo/', 'root /foo/%2E');
    u = URI('/foo/%2E%2E%2Fbar').normalizePath();
    equal(u.path(), '/foo/..%2Fbar', 'root /foo/%2E%2E%2Fbar');
    u = URI('../../../../../www/common/js/app/../../../../www_test/common/js/app/views/view-test.html');
    u.normalize();
    equal(u.path(), '../../../../../www_test/common/js/app/views/view-test.html', 'parent relative');
    u = URI('/../../../../../www/common/js/app/../../../../www_test/common/js/app/views/view-test.html');
    u.normalize();
    equal(u.path(), '/www_test/common/js/app/views/view-test.html', 'parent absolute');
    // URNs
    u = URI('urn:people:authors:poets:Shel Silverstein');
    u.normalize();
    equal(u.path(), 'people:authors:poets:Shel%20Silverstein');
    u = URI('urn:people:authors:philosophers:Søren Kierkegaard');
    u.normalize();
    equal(u.path(), 'people:authors:philosophers:S%C3%B8ren%20Kierkegaard');
    // URNs path separator preserved
    u = URI('urn:games:cards:Magic%3A the Gathering');
    u.normalize();
    equal(u.path(), 'games:cards:Magic%3A%20the%20Gathering');
  });
  test('normalizeQuery', function() {
    var u = new URI('http://example.org/foobar.html?');
    u.normalizeQuery();
    equal(u+'', 'http://example.org/foobar.html', 'dropping empty query sign');
    u.query('?&foo=bar&&baz=bam&').normalizeQuery();
    equal(u.query(), 'foo=bar&baz=bam', 'bad query resolution');
    u.query('?&foo=bar&&baz=bam&&baz=bau&').normalizeQuery();
    equal(u.query(), 'foo=bar&baz=bam&baz=bau', 'bad query resolution');
    u.query('?&foo=bar&foo=bar').normalizeQuery();
    equal(u.query(), 'foo=bar', 'duplicate key=value resolution');
  });
  test('normalizeFragment', function() {
    var u = new URI('http://example.org/foobar.html#');
    u.normalizeFragment();
    equal(u+'', 'http://example.org/foobar.html', 'dropping empty fragment sign');
  });
  test('readable', function() {
    var u = new URI('http://foo:bar@www.xn--exmple-cua.org/hello%20world/ä.html?foo%5B%5D=b+är#fragment');
    equal(u.readable(), 'http://www.exämple.org/hello world/ä.html?foo[]=b är#fragment', 'readable URL');
  });
  module('resolving URLs');
  test('absoluteTo', function() {
    // this being '../bar/baz.html?foo=bar'
    // base being 'http://example.org/foo/other/file.html'
    // return being http://example.org/foo/bar/baz.html?foo=bar'
    var tests = [{
        name: 'relative resolve',
        url: 'relative/path?blubber=1#hash1',
        base: 'http://www.example.org/path/to/file?some=query#hash',
        result: 'http://www.example.org/path/to/relative/path?blubber=1#hash1'
      }, {
        name: 'absolute resolve',
        url: '/absolute/path?blubber=1#hash1',
        base: 'http://www.example.org/path/to/file?some=query#hash',
        result: 'http://www.example.org/absolute/path?blubber=1#hash1'
      }, {
        name: 'relative resolve full URL',
        url: 'relative/path?blubber=1#hash3',
        base: 'http://user:pass@www.example.org:1234/path/to/file?some=query#hash',
        result: 'http://user:pass@www.example.org:1234/path/to/relative/path?blubber=1#hash3'
      }, {
        name: 'absolute resolve full URL',
        url: '/absolute/path?blubber=1#hash3',
        base: 'http://user:pass@www.example.org:1234/path/to/file?some=query#hash',
        result: 'http://user:pass@www.example.org:1234/absolute/path?blubber=1#hash3'
      }, {
        name: 'absolute resolve full URL without scheme',
        url: '//user:pass@www.example.org:1234/path/to/file?some=query#hash',
        base: 'proto://user:pass@www.example.org:1234/path/to/file?some=query#hash',
        result: 'proto://user:pass@www.example.org:1234/path/to/file?some=query#hash'
      }, {
        name: 'path-relative resolve',
        url: './relative/path?blubber=1#hash3',
        base: 'http://user:pass@www.example.org:1234/path/to/file?some=query#hash',
        result: 'http://user:pass@www.example.org:1234/path/to/relative/path?blubber=1#hash3'
      }, {
        name: 'path-relative parent resolve',
        url: '../relative/path?blubber=1#hash3',
        base: 'http://user:pass@www.example.org:1234/path/to/file?some=query#hash',
        result: 'http://user:pass@www.example.org:1234/path/relative/path?blubber=1#hash3'
      }, {
        name: 'path-relative path resolve',
        url: './relative/path?blubber=1#hash3',
        base: '/path/to/file?some=query#hash',
        result: '/path/to/relative/path?blubber=1#hash3'
      }, {
        name: 'path-relative path resolve 2',
        url: 'tofile',
        base: '/path/to/file',
        result: '/path/to/tofile'
      }, {
        name: 'path-relative path-root resolve',
        url: 'tofile',
        base: '/file',
        result: '/tofile'
      }, {
        name: 'path-relative parent path resolve',
        url: '../relative/path?blubber=1#hash3',
        base: '/path/to/file?some=query#hash',
        result: '/path/relative/path?blubber=1#hash3'
      }, {
        name: 'fragment absolute url',
        url: '#hash3',
        base: '/path/to/file?some=query#hash',
        result: '/path/to/file?some=query#hash3'
      }, {
        name: 'fragment relative url',
        url: '#hash3',
        base: 'path/to/file',
        result: 'path/to/file#hash3'
      }, {
        name: 'relative path - urljoin',
        url: 'the_relative_url',
        base: 'rel/path/',
        result: 'rel/path/the_relative_url'
      }, {
        name: 'relative path file - urljoin',
        url: 'the_relative_url',
        base: 'rel/path/something',
        result: 'rel/path/the_relative_url'
      }, {
        name: 'relative parent path file - urljoin',
        url: '../the_relative_url',
        base: 'rel/path/',
        result: 'rel/the_relative_url'
      }, {
        name: 'relative root path file - urljoin',
        url: '/the_relative_url',
        base: 'rel/path/',
        result: '/the_relative_url'
      }, {
        name: 'relative root file - urljoin',
        url: '/the_relative_url',
        base: 'http://example.com/rel/path/',
        result: 'http://example.com/the_relative_url'
      }, {
        name: 'absolute passthru - urljoin',
        url: 'http://github.com//the_relative_url',
        base: 'http://example.com/foo/bar',
        result: 'http://github.com//the_relative_url'
      }, {
        name: 'file paths - urljoin',
        url: 'anotherFile',
        base: 'aFile',
        result: 'anotherFile'
      }
    ];
    for (var i = 0, t; (t = tests[i]); i++) {
      var u = new URI(t.url),
        r = u.absoluteTo(t.base);
      equal(r + '', t.result, t.name);
    }
  });
  test('absoluteTo - RFC3986 reference resolution', function() {
    // http://tools.ietf.org/html/rfc3986#section-5.4
    var base = 'http://a/b/c/d;p?q';
    var map = {
      // normal
      // 'g:h'       :  'g:h', // identified as URN
      'g'       :  'http://a/b/c/g',
      './g'       :  'http://a/b/c/g',
      'g/'      :  'http://a/b/c/g/',
      '/g'      :  'http://a/g',
      '//g'       :  'http://g/', // added trailing /
      '?y'      :  'http://a/b/c/d;p?y',
      'g?y'       :  'http://a/b/c/g?y',
      '#s'      :  'http://a/b/c/d;p?q#s',
      'g#s'       :  'http://a/b/c/g#s',
      'g?y#s'     :  'http://a/b/c/g?y#s',
      ';x'      :  'http://a/b/c/;x',
      'g;x'       :  'http://a/b/c/g;x',
      'g;x?y#s'     :  'http://a/b/c/g;x?y#s',
      ''        :  'http://a/b/c/d;p?q',
      '.'       :  'http://a/b/c/',
      './'      :  'http://a/b/c/',
      '..'      :  'http://a/b/',
      '../'       :  'http://a/b/',
      '../g'      :  'http://a/b/g',
      '../..'     :  'http://a/',
      '../../'    :  'http://a/',
      '../../g'     :  'http://a/g',
      // abnormal
      '../../../g'  :  'http://a/g',
      '../../../../g' :  'http://a/g'
    };
    for (var key in map) {
      var u = new URI(key),
        r = u.absoluteTo(base);
      equal(r + '', map[key], 'resolution "' + key + '"');
    }
  });
  test('relativeTo', function() {
    var tests = [{
        name: 'same parent',
        url: '/relative/path?blubber=1#hash1',
        base: '/relative/file?some=query#hash',
        result: 'path?blubber=1#hash1'
      }, {
        name: 'direct parent',
        url: '/relative/path?blubber=1#hash1',
        base: '/relative/sub/file?some=query#hash',
        result: '../path?blubber=1#hash1'
      }, {
        name: 'second parent',
        url: '/relative/path?blubber=1#hash1',
        base: '/relative/sub/sub/file?some=query#hash',
        result: '../../path?blubber=1#hash1'
      }, {
        name: 'third parent',
        url: '/relative/path?blubber=1#hash1',
        base: '/relative/sub/foo/sub/file?some=query#hash',
        result: '../../../path?blubber=1#hash1'
      }, {
        name: 'parent top level',
        url: '/relative/path?blubber=1#hash1',
        base: '/path/to/file?some=query#hash',
        result: '../../relative/path?blubber=1#hash1'
      }, {
        name: 'descendant',
        url: '/base/path/with/subdir/inner.html',
        base: '/base/path/top.html',
        result: 'with/subdir/inner.html'
      }, {
        name: 'same directory',
        url: '/path/',
        base: '/path/top.html',
        result: './'
      }, {
        name: 'absolute /',
        url: 'http://example.org/foo/bar/bat',
        base: 'http://example.org/',
        result: 'foo/bar/bat'
      }, {
        name: 'absolute /foo',
        url: 'http://example.org/foo/bar/bat',
        base: 'http://example.org/foo',
        result: 'foo/bar/bat'
      }, {
        name: 'absolute /foo/',
        url: 'http://example.org/foo/bar/bat',
        base: 'http://example.org/foo/',
        result: 'bar/bat'
      }, {
        name: 'same scheme',
        url: 'http://example.org/foo/bar/bat',
        base: 'http://example.com/foo/',
        result: '//example.org/foo/bar/bat'
      }, {
        name: 'different scheme',
        url: 'http://example.org/foo/bar',
        base: 'https://example.org/foo/',
        result: 'http://example.org/foo/bar'
      }, {
        name: 'base with no scheme or host',
        url: 'http://example.org/foo/bar',
        base: '/foo/',
        result: 'http://example.org/foo/bar'
      }, {
        name: 'base with no scheme',
        url: 'http://example.org/foo/bar',
        base: '//example.org/foo/bar',
        result: 'http://example.org/foo/bar'
      }, {
        name: 'denormalized base',
        url: '/foo/bar/bat',
        base: '/foo/./bar/',
        result: 'bat'
      }, {
        name: 'denormalized url',
        url: '/foo//bar/bat',
        base: '/foo/bar/',
        result: 'bat'
      }, {
        name: 'credentials',
        url: 'http://user:pass@example.org/foo/bar',
        base: 'http://example.org/foo/',
        result: '//user:pass@example.org/foo/bar'
      }, {
        name: 'base credentials',
        url: 'http://example.org/foo/bar',
        base: 'http://user:pass@example.org/foo/bar',
        result: '//example.org/foo/bar'
      }, {
        name: 'same credentials different host',
        url: 'http://user:pass@example.org/foo/bar',
        base: 'http://user:pass@example.com/foo/bar',
        result: '//user:pass@example.org/foo/bar'
      }, {
        name: 'different port 1',
        url: 'http://example.org/foo/bar',
        base: 'http://example.org:8080/foo/bar',
        result: '//example.org/foo/bar'
      }, {
        name: 'different port 2',
        url: 'http://example.org:8081/foo/bar',
        base: 'http://example.org:8080/foo/bar',
        result: '//example.org:8081/foo/bar'
      }, {
        name: 'different port 3',
        url: 'http://example.org:8081/foo/bar',
        base: 'http://example.org/foo/bar',
        result: '//example.org:8081/foo/bar'
      }, {
        name: 'same path - fragment',
        url: 'http://www.example.com:8080/dir/file#abcd',
        base: 'http://www.example.com:8080/dir/file',
        result: '#abcd'
      }, {
        name: 'same path - query',
        url: 'http://www.example.com:8080/dir/file?abcd=123',
        base: 'http://www.example.com:8080/dir/file',
        result: '?abcd=123'
      }, {
        name: 'same path - query and fragment',
        url: 'http://www.example.com:8080/dir/file?abcd=123#alpha',
        base: 'http://www.example.com:8080/dir/file',
        result: '?abcd=123#alpha'
      }, {
        name: 'already relative',
        url: 'foo/bar',
        base: '/foo/',
        'throws': true
      }, {
        name: 'relative base',
        url: '/foo/bar',
        base: 'foo/',
        'throws': true
      }
    ];
    for (var i = 0, t; (t = tests[i]); i++) {
      var u = new URI(t.url),
        b = new URI(t.base),
        caught = false;
      var r;
      try {
        r = u.relativeTo(b);
      } catch (e) {
        caught = true;
      }
      /*jshint sub:true */
      if (t['throws']) {
      /*jshint sub:false */
        ok(caught, t.name + ' should throw exception');
      } else {
        ok(!caught, t.name + ' should not throw exception');
        equal(r + '', t.result, t.name);
        var a = r.absoluteTo(t.base);
        var n = u.clone().normalize();
        equal(a.toString(), n.toString(), t.name + ' reversed');
      }
    }
    equal('b/c',
      new URI('http://example.org/a/b/c')
        .scheme('')
        .authority('')
        .relativeTo('/a/')
        .toString(),
      'bug #103');
    equal('b/c',
      new URI('//example.org/a/b/c')
        .authority('')
        .relativeTo('/a/')
        .toString(),
      'bug #103 (2)');
  });
  module('static helpers');
  test('withinString', function() {
    /*jshint laxbreak: true */
    var source = 'Hello www.example.com,\n'
      + 'http://google.com is a search engine, like http://www.bing.com\n'
      + 'http://exämple.org/foo.html?baz=la#bumm is an IDN URL,\n'
      + 'http://123.123.123.123/foo.html is IPv4 and http://fe80:0000:0000:0000:0204:61ff:fe9d:f156/foobar.html is IPv6.\n'
      + 'links can also be in parens (http://example.org) or quotes »http://example.org«.';
    var expected = 'Hello www.example.com,\n'
      + 'http://google.com is a search engine, like http://www.bing.com\n'
      + 'http://exämple.org/foo.html?baz=la#bumm is an IDN URL,\n'
      + 'http://123.123.123.123/foo.html is IPv4 and http://fe80:0000:0000:0000:0204:61ff:fe9d:f156/foobar.html is IPv6.\n'
      + 'links can also be in parens (http://example.org) or quotes »http://example.org«.';
    /*jshint laxbreak: false */
    var result = URI.withinString(source, function(url) {
      return '' + url + '';
    });
    equal(result, expected, 'in string URI identification');
  });
  test('withinString - ignore', function() {
    var decorate = function(url) {
      return '' + url + '';
    };
    /*jshint laxbreak: true */
    var source = 'Hello www.example.com,\n'
      + 'proto://example.org/foo.html?baz=la#bumm is an URL.\n';
    var expected = 'Hello www.example.com,\n'
      + 'proto://example.org/foo.html?baz=la#bumm is an URL.\n';
    /*jshint laxbreak: false */
    var result = URI.withinString(source, decorate, {ignore: /^proto:/i});
    equal(result, expected, 'filtered in string URI identification');
  });
  test('withinString - ignoreHtml', function() {
    var decorate = function(url) {
      return '' + url + '';
    };
    /*jshint laxbreak: true */
    var source = 'Hello www.example.com,\n'
      + '.\n'
      + ' is an URL.\n'
      + ' is an URL.\n';
    /*jshint laxbreak: false */
    var result = URI.withinString(source, decorate, {ignoreHtml: true});
    equal(result, expected, 'filtered in string URI identification');
  });
  test('withinString - capture only', function() {
    /*jshint laxbreak: true */
    var source = 'Hello www.example.com,\n'
      + 'http://google.com is a search engine, like http://www.bing.com\n'
      + 'http://exämple.org/foo.html?baz=la#bumm is an IDN URL,\n'
      + 'http://123.123.123.123/foo.html is IPv4 and http://fe80:0000:0000:0000:0204:61ff:fe9d:f156/foobar.html is IPv6.\n'
      + 'links can also be in parens (http://example.org) or quotes »http://example.org«.';
    var expected = [
      'www.example.com',
      'http://google.com',
      'http://www.bing.com',
      'http://exämple.org/foo.html?baz=la#bumm',
      'http://123.123.123.123/foo.html',
      'http://fe80:0000:0000:0000:0204:61ff:fe9d:f156/foobar.html',
      'http://example.org',
      'http://example.org'
    ];
    /*jshint laxbreak: false */
    var links = [];
    var result = URI.withinString(source, function(url) {
      links.push(url);
    });
    deepEqual(links, expected, 'urls extracted');
    equal(result, source, 'source not modified');
  });
  test('noConflict', function() {
    var actual_lib = URI; // actual library; after loading, before noConflict()
    var unconflicted = URI.noConflict();
    strictEqual( unconflicted, actual_lib, 'noConflict() returns the URI object' );
    strictEqual( URI, URI_pre_lib, 'noConflict() restores the `URI` variable' );
    // restore for other tests
    window.URI = actual_lib;
  });
  test('noConflict(removeAll=true)', function() {
    var actual = {
      URI:        URI,
      URITemplate:    URITemplate,
      IPv6:         IPv6,
      SecondLevelDomains: SecondLevelDomains
    };
    var unconflicted = URI.noConflict(true);
    deepEqual( unconflicted, actual, 'noConflict(true) returns the { URI, URITemplate, IPv6, SecondLevelDomains } object' );
    strictEqual( URI,        URI_pre_lib,        'noConflict(true) restores the `URI` variable' );
    strictEqual( URITemplate,    URITemplate_pre_lib,    'noConflict(true) restores the `URITemplate` variable' );
    strictEqual( IPv6,         IPv6_pre_lib,         'noConflict(true) restores the `IPv6` variable' );
    strictEqual( SecondLevelDomains, SecondLevelDomains_pre_lib, 'noConflict(true) restores the `SecondLevelDomains` variable' );
    // restore for other tests
    window.URI        = actual.URI;
    window.URITemplate    = actual.URITemplate;
    window.IPv6         = actual.IPv6;
    window.SecondLevelDomains = actual.SecondLevelDomains;
  });
  test('joinPaths', function() {
    var result;
    
    result = URI.joinPaths('/a/b', '/c', 'd', '/e').toString();
    equal(result, '/a/b/c/d/e', 'absolute paths');
    result = URI.joinPaths('a/b', 'http://example.com/c', new URI('d/'), '/e').toString();
    equal(result, 'a/b/c/d/e', 'relative path');
    result = URI.joinPaths('/a/').toString();
    equal(result, '/a/', 'single absolute directory');
    result = URI.joinPaths('/a').toString();
    equal(result, '/a', 'single absolute segment');
    result = URI.joinPaths('a').toString();
    equal(result, 'a', 'single relative segment');
    result = URI.joinPaths('').toString();
    equal(result, '', 'empty string');
    result = URI.joinPaths().toString();
    equal(result, '', 'no argument');
    result = URI.joinPaths('', 'a', '', '', 'b').toString();
    equal(result, '/a/b', 'leading empty segment');
    result = URI.joinPaths('a', '', '', 'b', '', '').toString();
    equal(result, 'a/b/', 'trailing empty segment');
  });
  module('comparing URLs');
  test('equals', function() {
    var u = new URI('http://example.org/foo/bar.html?foo=bar&hello=world&hello=mars#fragment'),
      e = [
        'http://example.org/foo/../foo/bar.html?foo=bar&hello=world&hello=mars#fragment',
        'http://exAmple.org/foo/bar.html?foo=bar&hello=world&hello=mars#fragment',
        'http://exAmple.org:80/foo/bar.html?foo=bar&hello=world&hello=mars#fragment',
        'http://example.org/foo/bar.html?foo=bar&hello=mars&hello=world#fragment',
        'http://example.org/foo/bar.html?hello=mars&hello=world&foo=barfragment'
      ],
      d = [
        'http://example.org/foo/../bar.html?foo=bar&hello=world&hello=mars#fragment',
        'http://example.org/foo/bar.html?foo=bar&hello=world&hello=mars#frAgment',
        'http://example.org/foo/bar.html?foo=bar&hello=world&hello=mArs#fragment',
        'http://example.org/foo/bar.hTml?foo=bar&hello=world&hello=mars#fragment',
        'http://example.org:8080/foo/bar.html?foo=bar&hello=world&hello=mars#fragment',
        'http://user:pass@example.org/foo/bar.html?foo=bar&hello=world&hello=mars#fragment',
        'ftp://example.org/foo/bar.html?foo=bar&hello=world&hello=mars#fragment',
        'http://example.org/foo/bar.html?foo=bar&hello=world&hello=mars&hello=jupiter#fragment'
      ],
      i, c;
    for (i = 0; (c = e[i]); i++) {
      equal(u.equals(c), true, 'equality ' + i);
    }
    for (i = 0; (c = d[i]); i++) {
      equal(u.equals(c), false, 'different ' + i);
    }
  });
  module('Charset');
  test('iso8859', function() {
    var u = new URI('/ä.html');
    u.normalizePath();
    equal(u.path(), '/%C3%A4.html', 'Unicode');
    URI.iso8859();
    u = new URI('/ä.html');
    u.normalizePath();
    equal(u.path(), '/%E4.html', 'ISO8859');
    u.path('/ö.html');
    equal(u.path(), '/%F6.html', 'ISO8859');
    URI.unicode();
    u = new URI('/ä.html');
    u.normalizePath();
    equal(u.path(), '/%C3%A4.html', 'Unicode again');
    u = new URI('/ä.html');
    u.normalizePath();
    equal(u.path(), '/%C3%A4.html', 'convert unicode start');
    u.iso8859();
    equal(u.path(), '/%E4.html', 'convert iso8859');
    u.unicode();
    equal(u.path(), '/%C3%A4.html', 'convert unicode');
  });
  test('bad charset in QueryString', function() {
    var uri = new URI('http://www.google.com.hk/search?q=pennytel%20downloads&sa=%20%CB%D1%20%CB%F7%20&forid=1&prog=aff&ie=GB2312&oe=GB2312&safe=active&source=sdo_sb_html&hl=zh-CN');
    var data = uri.query(true);
    equal(data.sa, '%20%CB%D1%20%CB%F7%20', 'undecodable value returned');
    equal(data.forid, '1', 'decodable value returned');
    uri.normalizeQuery();
    data = uri.query(true);
    equal(data.sa, '%20%CB%D1%20%CB%F7%20', 'undecodable value returned');
    equal(data.forid, '1', 'decodable value returned');
  });
  module('Encoding');
  test('decode malformed URI', function() {
    try {
      decodeURIComponent('%%20');
      ok(false, 'decodeURIComponent() must throw URIError: URI malformed');
    } catch(e) {}
    try {
      URI.decode('%%20');
      ok(false, 'URI.decode() must throw URIError: URI malformed');
    } catch(e) {}
    equal(URI.decodeQuery('%%20'), '%%20', 'malformed URI component returned');
    equal(URI.decodePathSegment('%%20'), '%%20', 'malformed URI component returned');
    equal(URI.decodeUrnPathSegment('%%20'), '%%20', 'malformed URN component returned');
  });
  test('encodeQuery', function() {
    var escapeQuerySpace = URI.escapeQuerySpace;
    URI.escapeQuerySpace = true;
    equal(URI.encodeQuery(' '), '+');
    equal(URI.encode(' '), '%20');
    URI.escapeQuerySpace = false;
    equal(URI.encodeQuery(' '), '%20');
    equal(URI.encode(' '), '%20');
    URI.escapeQuerySpace = escapeQuerySpace;
  });
  test('decodeQuery', function() {
    var escapeQuerySpace = URI.escapeQuerySpace;
    URI.escapeQuerySpace = true;
    equal(URI.decodeQuery('+'), ' ');
    equal(URI.decodeQuery('%20'), ' ');
    equal(URI.decode('%20'), ' ');
    equal(URI.decode('+'), '+');
    URI.escapeQuerySpace = false;
    equal(URI.decodeQuery('+'), '+');
    equal(URI.decodeQuery('%20'), ' ');
    equal(URI.decode('%20'), ' ');
    equal(URI.decode('+'), '+');
    URI.escapeQuerySpace = escapeQuerySpace;
  });
  test('encodeReserved', function() {
    equal(URI.encodeReserved('ä:/?#[]@!$&\'()*+,;='), '%C3%A4:/?#[]@!$&\'()*+,;=');
  });
})();