2018-07-23 14:46:30 -04:00
|
|
|
// Copyright 2018 the Deno authors. All rights reserved. MIT license.
|
2018-07-06 11:20:35 -04:00
|
|
|
import {
|
|
|
|
assert,
|
|
|
|
log,
|
|
|
|
createResolvable,
|
|
|
|
Resolvable,
|
2018-08-19 21:04:27 +02:00
|
|
|
typedArrayToArrayBuffer,
|
2018-08-21 12:10:09 -04:00
|
|
|
notImplemented
|
2018-07-06 11:20:35 -04:00
|
|
|
} from "./util";
|
2018-08-15 20:57:36 -04:00
|
|
|
import { flatbuffers } from "flatbuffers";
|
2018-09-09 18:54:42 -04:00
|
|
|
import { sendAsync } from "./dispatch";
|
2018-09-09 19:21:22 -04:00
|
|
|
import * as fbs from "gen/msg_generated";
|
2018-08-15 20:57:36 -04:00
|
|
|
import {
|
|
|
|
Headers,
|
|
|
|
Request,
|
|
|
|
Response,
|
|
|
|
Blob,
|
|
|
|
RequestInit,
|
2018-09-12 23:46:42 +04:30
|
|
|
HeadersInit,
|
2018-08-15 20:57:36 -04:00
|
|
|
FormData
|
2018-09-14 22:43:38 +04:30
|
|
|
} from "./dom_types";
|
2018-08-15 20:57:36 -04:00
|
|
|
import { TextDecoder } from "./text_encoding";
|
2018-09-14 22:26:37 +04:30
|
|
|
import { DenoBlob } from "./blob";
|
2018-08-15 20:57:36 -04:00
|
|
|
|
2018-09-12 23:46:42 +04:30
|
|
|
interface Header {
|
|
|
|
name: string;
|
|
|
|
value: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
export class DenoHeaders implements Headers {
|
|
|
|
private readonly headerList: Header[] = [];
|
|
|
|
|
|
|
|
constructor(init?: HeadersInit) {
|
|
|
|
if (init) {
|
|
|
|
this._fill(init);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private _append(header: Header): void {
|
|
|
|
// TODO(qti3e) Check header based on the fetch spec.
|
|
|
|
this._appendToHeaderList(header);
|
|
|
|
}
|
|
|
|
|
|
|
|
private _appendToHeaderList(header: Header): void {
|
|
|
|
const lowerCaseName = header.name.toLowerCase();
|
|
|
|
for (let i = 0; i < this.headerList.length; ++i) {
|
|
|
|
if (this.headerList[i].name.toLowerCase() === lowerCaseName) {
|
|
|
|
header.name = this.headerList[i].name;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.headerList.push(header);
|
|
|
|
}
|
|
|
|
|
|
|
|
private _fill(init: HeadersInit): void {
|
|
|
|
if (Array.isArray(init)) {
|
|
|
|
for (let i = 0; i < init.length; ++i) {
|
|
|
|
const header = init[i];
|
|
|
|
if (header.length !== 2) {
|
|
|
|
throw new TypeError("Failed to construct 'Headers': Invalid value");
|
|
|
|
}
|
|
|
|
this._append({
|
|
|
|
name: header[0],
|
|
|
|
value: header[1]
|
|
|
|
});
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (const key in init) {
|
|
|
|
this._append({
|
|
|
|
name: key,
|
|
|
|
value: init[key]
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-15 20:57:36 -04:00
|
|
|
append(name: string, value: string): void {
|
2018-09-12 23:46:42 +04:30
|
|
|
this._appendToHeaderList({ name, value });
|
2018-08-15 20:57:36 -04:00
|
|
|
}
|
2018-09-12 23:46:42 +04:30
|
|
|
|
2018-08-15 20:57:36 -04:00
|
|
|
delete(name: string): void {
|
|
|
|
assert(false, "Implement me");
|
|
|
|
}
|
|
|
|
get(name: string): string | null {
|
2018-09-12 23:46:42 +04:30
|
|
|
for (const header of this.headerList) {
|
|
|
|
if (header.name.toLowerCase() === name.toLowerCase()) {
|
|
|
|
return header.value;
|
|
|
|
}
|
|
|
|
}
|
2018-08-15 20:57:36 -04:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
has(name: string): boolean {
|
|
|
|
assert(false, "Implement me");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
set(name: string, value: string): void {
|
|
|
|
assert(false, "Implement me");
|
|
|
|
}
|
|
|
|
forEach(
|
|
|
|
callbackfn: (value: string, key: string, parent: Headers) => void,
|
|
|
|
// tslint:disable-next-line:no-any
|
|
|
|
thisArg?: any
|
|
|
|
): void {
|
|
|
|
assert(false, "Implement me");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-06 11:20:35 -04:00
|
|
|
class FetchResponse implements Response {
|
2018-09-05 22:13:36 -04:00
|
|
|
readonly url: string = "";
|
2018-07-06 11:20:35 -04:00
|
|
|
body: null;
|
|
|
|
bodyUsed = false; // TODO
|
|
|
|
statusText = "FIXME"; // TODO
|
|
|
|
readonly type = "basic"; // TODO
|
|
|
|
redirected = false; // TODO
|
2018-09-12 23:46:42 +04:30
|
|
|
headers: DenoHeaders;
|
2018-08-15 20:57:36 -04:00
|
|
|
readonly trailer: Promise<Headers>;
|
2018-07-06 11:20:35 -04:00
|
|
|
//private bodyChunks: Uint8Array[] = [];
|
|
|
|
private first = true;
|
2018-08-15 20:57:36 -04:00
|
|
|
private bodyWaiter: Resolvable<ArrayBuffer>;
|
2018-07-06 11:20:35 -04:00
|
|
|
|
2018-09-12 23:46:42 +04:30
|
|
|
constructor(
|
|
|
|
readonly status: number,
|
|
|
|
readonly body_: ArrayBuffer,
|
|
|
|
headersList: Array<[string, string]>
|
|
|
|
) {
|
2018-08-15 20:57:36 -04:00
|
|
|
this.bodyWaiter = createResolvable();
|
|
|
|
this.trailer = createResolvable();
|
2018-09-12 23:46:42 +04:30
|
|
|
this.headers = new DenoHeaders(headersList);
|
2018-09-05 22:13:36 -04:00
|
|
|
setTimeout(() => {
|
|
|
|
this.bodyWaiter.resolve(body_);
|
|
|
|
}, 0);
|
2018-07-06 11:20:35 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
arrayBuffer(): Promise<ArrayBuffer> {
|
|
|
|
return this.bodyWaiter;
|
|
|
|
}
|
|
|
|
|
2018-08-20 21:03:11 -04:00
|
|
|
async blob(): Promise<Blob> {
|
2018-09-14 22:26:37 +04:30
|
|
|
const arrayBuffer = await this.arrayBuffer();
|
|
|
|
return new DenoBlob([arrayBuffer], {
|
|
|
|
type: this.headers.get("content-type") || ""
|
|
|
|
});
|
2018-07-06 11:20:35 -04:00
|
|
|
}
|
|
|
|
|
2018-08-20 21:03:11 -04:00
|
|
|
async formData(): Promise<FormData> {
|
2018-08-19 21:04:27 +02:00
|
|
|
notImplemented();
|
2018-08-20 21:03:11 -04:00
|
|
|
return {} as FormData;
|
2018-07-06 11:20:35 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
async json(): Promise<object> {
|
|
|
|
const text = await this.text();
|
|
|
|
return JSON.parse(text);
|
|
|
|
}
|
|
|
|
|
|
|
|
async text(): Promise<string> {
|
|
|
|
const ab = await this.arrayBuffer();
|
|
|
|
const decoder = new TextDecoder("utf-8");
|
|
|
|
return decoder.decode(ab);
|
|
|
|
}
|
|
|
|
|
|
|
|
get ok(): boolean {
|
|
|
|
return 200 <= this.status && this.status < 300;
|
|
|
|
}
|
|
|
|
|
|
|
|
clone(): Response {
|
2018-08-19 21:04:27 +02:00
|
|
|
notImplemented();
|
2018-08-20 21:03:11 -04:00
|
|
|
return {} as Response;
|
2018-07-06 11:20:35 -04:00
|
|
|
}
|
|
|
|
|
2018-08-15 20:57:36 -04:00
|
|
|
onHeader?: (res: FetchResponse) => void;
|
|
|
|
onError?: (error: Error) => void;
|
2018-07-06 11:20:35 -04:00
|
|
|
|
2018-09-05 22:13:36 -04:00
|
|
|
onMsg(base: fbs.Base) {
|
|
|
|
/*
|
2018-08-15 20:57:36 -04:00
|
|
|
const error = base.error();
|
|
|
|
if (error != null) {
|
|
|
|
assert(this.onError != null);
|
|
|
|
this.onError!(new Error(error));
|
2018-07-06 11:20:35 -04:00
|
|
|
return;
|
|
|
|
}
|
2018-09-05 22:13:36 -04:00
|
|
|
*/
|
2018-07-06 11:20:35 -04:00
|
|
|
|
|
|
|
if (this.first) {
|
|
|
|
this.first = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-05 22:13:36 -04:00
|
|
|
export async function fetch(
|
2018-07-06 11:20:35 -04:00
|
|
|
input?: Request | string,
|
|
|
|
init?: RequestInit
|
|
|
|
): Promise<Response> {
|
2018-09-05 22:13:36 -04:00
|
|
|
const url = input as string;
|
|
|
|
log("dispatch FETCH_REQ", url);
|
|
|
|
|
|
|
|
// Send FetchReq message
|
|
|
|
const builder = new flatbuffers.Builder();
|
|
|
|
const url_ = builder.createString(url);
|
|
|
|
fbs.FetchReq.startFetchReq(builder);
|
|
|
|
fbs.FetchReq.addUrl(builder, url_);
|
|
|
|
const resBase = await sendAsync(
|
|
|
|
builder,
|
|
|
|
fbs.Any.FetchReq,
|
|
|
|
fbs.FetchReq.endFetchReq(builder)
|
|
|
|
);
|
|
|
|
|
|
|
|
// Decode FetchRes
|
|
|
|
assert(fbs.Any.FetchRes === resBase.msgType());
|
|
|
|
const msg = new fbs.FetchRes();
|
|
|
|
assert(resBase.msg(msg) != null);
|
|
|
|
|
|
|
|
const status = msg.status();
|
|
|
|
const bodyArray = msg.bodyArray();
|
|
|
|
assert(bodyArray != null);
|
|
|
|
const body = typedArrayToArrayBuffer(bodyArray!);
|
|
|
|
|
2018-09-12 23:46:42 +04:30
|
|
|
const headersList: Array<[string, string]> = [];
|
|
|
|
const len = msg.headerKeyLength();
|
|
|
|
for (let i = 0; i < len; ++i) {
|
|
|
|
const key = msg.headerKey(i);
|
|
|
|
const value = msg.headerValue(i);
|
|
|
|
headersList.push([key, value]);
|
|
|
|
}
|
|
|
|
|
|
|
|
const response = new FetchResponse(status, body, headersList);
|
2018-09-05 22:13:36 -04:00
|
|
|
return response;
|
2018-07-06 11:20:35 -04:00
|
|
|
}
|