2024-02-19 19:12:42 +01:00
|
|
|
/**
|
|
|
|
* TODOC
|
2024-02-19 19:26:15 +01:00
|
|
|
* TODO: document so we can improve this
|
2024-02-22 21:23:39 +01:00
|
|
|
* @param {*} url
|
|
|
|
* @returns
|
2024-02-19 19:12:42 +01:00
|
|
|
*/
|
2023-07-15 01:48:36 +00:00
|
|
|
function parseUrl(url) {
|
|
|
|
// XXX: Hack.
|
2024-02-24 11:09:34 -05:00
|
|
|
let match = url.match(new RegExp('(\\w+)://([^/:]+)(?::(\\d+))?(.*)'));
|
2023-07-15 01:48:36 +00:00
|
|
|
return {
|
|
|
|
protocol: match[1],
|
|
|
|
host: match[2],
|
2023-07-16 21:04:48 +00:00
|
|
|
path: match[4],
|
2024-02-24 11:09:34 -05:00
|
|
|
port: match[3] ? parseInt(match[3]) : match[1] == 'http' ? 80 : 443,
|
2023-07-15 01:48:36 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2024-02-19 19:12:42 +01:00
|
|
|
/**
|
|
|
|
* TODOC
|
2024-02-22 21:23:39 +01:00
|
|
|
* @param {*} data
|
|
|
|
* @returns
|
2024-02-19 19:12:42 +01:00
|
|
|
*/
|
2023-07-15 01:48:36 +00:00
|
|
|
function parseResponse(data) {
|
|
|
|
let firstLine;
|
|
|
|
let headers = {};
|
|
|
|
while (true) {
|
|
|
|
let endLine = data.indexOf('\r\n');
|
|
|
|
let line = data.substring(0, endLine);
|
2023-07-16 21:04:48 +00:00
|
|
|
data = data.substring(endLine + 2);
|
|
|
|
if (!line.length) {
|
2023-07-15 01:48:36 +00:00
|
|
|
break;
|
2023-07-16 21:04:48 +00:00
|
|
|
} else if (!firstLine) {
|
|
|
|
firstLine = line;
|
2023-07-15 01:48:36 +00:00
|
|
|
} else {
|
2024-02-24 11:09:34 -05:00
|
|
|
let colon = line.indexOf(':');
|
2023-07-15 01:48:36 +00:00
|
|
|
headers[line.substring(colon)] = line.substring(colon + 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return {body: data};
|
|
|
|
}
|
|
|
|
|
2024-02-19 19:12:42 +01:00
|
|
|
/**
|
|
|
|
* TODOC
|
2024-02-22 21:23:39 +01:00
|
|
|
* @param {*} url
|
|
|
|
* @param {*} options
|
|
|
|
* @param {*} allowed_hosts
|
|
|
|
* @returns
|
2024-02-19 19:12:42 +01:00
|
|
|
*/
|
2023-07-16 21:04:48 +00:00
|
|
|
export function fetch(url, options, allowed_hosts) {
|
2023-07-15 01:48:36 +00:00
|
|
|
let parsed = parseUrl(url);
|
2024-02-24 11:09:34 -05:00
|
|
|
return new Promise(function (resolve, reject) {
|
2023-07-23 01:12:11 +00:00
|
|
|
if ((allowed_hosts ?? []).indexOf(parsed.host) == -1) {
|
2023-07-16 21:04:48 +00:00
|
|
|
throw new Error(`fetch() request to host ${parsed.host} is not allowed.`);
|
|
|
|
}
|
2023-07-15 01:48:36 +00:00
|
|
|
let socket = new Socket();
|
2023-07-16 21:04:48 +00:00
|
|
|
let buffer = new Uint8Array(0);
|
2023-07-15 01:48:36 +00:00
|
|
|
|
2024-02-24 11:09:34 -05:00
|
|
|
return socket
|
|
|
|
.connect(parsed.host, parsed.port)
|
|
|
|
.then(function () {
|
|
|
|
socket.read(function (data) {
|
|
|
|
if (data && data.length) {
|
|
|
|
let newBuffer = new Uint8Array(buffer.length + data.length);
|
|
|
|
newBuffer.set(buffer, 0);
|
|
|
|
newBuffer.set(data, buffer.length);
|
|
|
|
buffer = newBuffer;
|
|
|
|
} else {
|
|
|
|
let result = parseHttpResponse(buffer);
|
|
|
|
if (!result) {
|
|
|
|
reject(new Exception('Parse failed.'));
|
|
|
|
}
|
|
|
|
if (typeof result == 'number') {
|
|
|
|
if (result == -2) {
|
|
|
|
reject('Incomplete request.');
|
|
|
|
} else {
|
|
|
|
reject('Bad request.');
|
|
|
|
}
|
|
|
|
} else if (typeof result == 'object') {
|
|
|
|
resolve({
|
|
|
|
body: buffer.slice(result.bytes_parsed),
|
|
|
|
status: result.status,
|
|
|
|
message: result.message,
|
|
|
|
headers: result.headers,
|
|
|
|
});
|
2023-07-23 01:12:11 +00:00
|
|
|
} else {
|
2024-02-24 11:09:34 -05:00
|
|
|
reject(new Exception('Unexpected parse result.'));
|
2023-07-23 01:12:11 +00:00
|
|
|
}
|
2024-02-24 11:09:34 -05:00
|
|
|
resolve(parseResponse(utf8Decode(buffer)));
|
2023-07-23 01:12:11 +00:00
|
|
|
}
|
2024-02-24 11:09:34 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
if (parsed.port == 443) {
|
|
|
|
return socket.startTls();
|
2023-07-15 01:48:36 +00:00
|
|
|
}
|
2024-02-24 11:09:34 -05:00
|
|
|
})
|
|
|
|
.then(function () {
|
|
|
|
let body =
|
|
|
|
typeof options?.body == 'string'
|
|
|
|
? utf8Encode(options.body)
|
|
|
|
: options.body || new Uint8Array(0);
|
|
|
|
let headers = utf8Encode(
|
|
|
|
`${options?.method ?? 'GET'} ${parsed.path} HTTP/1.0\r\nHost: ${parsed.host}\r\nConnection: close\r\nContent-Length: ${body.length}\r\n\r\n`
|
|
|
|
);
|
|
|
|
let fullRequest = new Uint8Array(headers.length + body.length);
|
|
|
|
fullRequest.set(headers, 0);
|
|
|
|
fullRequest.set(body, headers.length);
|
|
|
|
socket.write(fullRequest);
|
|
|
|
})
|
|
|
|
.catch(function (error) {
|
|
|
|
reject(error);
|
2023-07-15 01:48:36 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|