Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

new URL() - WHATWG URL API

I'm messing around with node and I'm trying to get an instance of the URL class (because of those handy properties). Like:

const { URL } = require('url');
(...)
http.createServer((request,response) => {
    let uri = new URL(request.url);
    (...)
}

But it fails with

TypeError [ERR_INVALID_URL]: Invalid URL: /

It's funny because

const url = require('url');
url.parse();

works. So I got curious about it. I understand this later method is older.

I'm developing locally so to send a request I use localhost:8000 in the browser.

How do I use the information in request to instantiate a new URL object?

Info and things I've looked on already:

node -v
v9.3.0

https://nodejs.org/api/url.html#url_class_url
https://stackoverflow.com/questions/44738065/uncaught-typeerror-url-is-not-a-constructor-using-whatwg-url-object-support-for
https://stackoverflow.com/questions/45047840/in-node-js-how-to-get-construct-url-string-from-whatwg-url-including-user-and-pa
https://stackoverflow.com/questions/47481784/does-url-parse-protect-against-in-a-url
https://stackoverflow.com/questions/17184791/node-js-url-parse-and-pathname-property
like image 843
Thom Avatar asked Jan 10 '18 21:01

Thom


1 Answers

As Joshua Wise pointed out on Github (https://github.com/nodejs/node/issues/12682), the problem is that request.url is NOT an absolute URL. It can be used as a path or relative URL, but it is missing the host and protocol.

Following the API documentation, you can construct a valid URL by supplying a base URL as the second argument. This is still awkward because the protocol is not easily available from request.headers. I simply assumed HTTP:

var baseURL = 'http://' + request.headers.host + '/';
var myURL = new URL(request.url, baseURL);

This is obviously not an ideal solution, but at least you can take advantage of the query string parsing. For example,

URL {
  href: 'http://localhost:8080/?key1=value1&key2=value2',
  origin: 'http://localhost:8080',
  protocol: 'http:',
  username: '',
  password: '',
  host: 'localhost:8080',
  hostname: 'localhost',
  port: '8080',
  pathname: '/',
  search: '?key1=value1&key2=value2',
  searchParams: URLSearchParams { 'key1' => 'value1', 'key2' => 'value2' },
  hash: '' }
like image 131
Jeshurun Hembd Avatar answered Nov 18 '22 16:11

Jeshurun Hembd