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 15:04:27 -04:00
|
|
|
typedArrayToArrayBuffer,
|
|
|
|
notImplemented,
|
2018-07-06 11:20:35 -04:00
|
|
|
} from "./util";
|
2018-08-15 20:57:36 -04:00
|
|
|
import { flatbuffers } from "flatbuffers";
|
|
|
|
import { libdeno } from "./globals";
|
|
|
|
import { deno as fbs } from "gen/msg_generated";
|
|
|
|
import {
|
|
|
|
Headers,
|
|
|
|
Request,
|
|
|
|
Response,
|
|
|
|
Blob,
|
|
|
|
RequestInit,
|
|
|
|
FormData
|
|
|
|
} from "./fetch_types";
|
|
|
|
import { TextDecoder } from "./text_encoding";
|
|
|
|
|
|
|
|
/** @internal */
|
|
|
|
export function onFetchRes(base: fbs.Base, msg: fbs.FetchRes) {
|
|
|
|
const id = msg.id();
|
|
|
|
const req = fetchRequests.get(id);
|
|
|
|
assert(req != null, `Couldn't find FetchRequest id ${id}`);
|
|
|
|
req!.onMsg(base, msg);
|
2018-07-06 11:20:35 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
const fetchRequests = new Map<number, FetchRequest>();
|
|
|
|
|
2018-08-15 20:57:36 -04:00
|
|
|
class DenoHeaders implements Headers {
|
|
|
|
append(name: string, value: string): void {
|
|
|
|
assert(false, "Implement me");
|
|
|
|
}
|
|
|
|
delete(name: string): void {
|
|
|
|
assert(false, "Implement me");
|
|
|
|
}
|
|
|
|
get(name: string): string | null {
|
|
|
|
assert(false, "Implement me");
|
|
|
|
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 {
|
|
|
|
readonly url: string;
|
|
|
|
body: null;
|
|
|
|
bodyUsed = false; // TODO
|
2018-08-15 20:57:36 -04:00
|
|
|
status = 0;
|
2018-07-06 11:20:35 -04:00
|
|
|
statusText = "FIXME"; // TODO
|
|
|
|
readonly type = "basic"; // TODO
|
|
|
|
redirected = false; // TODO
|
2018-08-15 20:57:36 -04:00
|
|
|
headers = new DenoHeaders();
|
|
|
|
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
|
|
|
|
|
|
|
constructor(readonly req: FetchRequest) {
|
|
|
|
this.url = req.url;
|
2018-08-15 20:57:36 -04:00
|
|
|
this.bodyWaiter = createResolvable();
|
|
|
|
this.trailer = createResolvable();
|
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-08-19 15:04:27 -04:00
|
|
|
notImplemented();
|
2018-08-20 21:03:11 -04:00
|
|
|
return {} as Blob;
|
2018-07-06 11:20:35 -04:00
|
|
|
}
|
|
|
|
|
2018-08-20 21:03:11 -04:00
|
|
|
async formData(): Promise<FormData> {
|
2018-08-19 15:04:27 -04: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 15:04:27 -04: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-08-15 20:57:36 -04:00
|
|
|
onMsg(base: fbs.Base, msg: fbs.FetchRes) {
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.first) {
|
|
|
|
this.first = false;
|
2018-08-15 20:57:36 -04:00
|
|
|
this.status = msg.status();
|
|
|
|
assert(this.onHeader != null);
|
|
|
|
this.onHeader!(this);
|
2018-07-06 11:20:35 -04:00
|
|
|
} else {
|
|
|
|
// Body message. Assuming it all comes in one message now.
|
2018-08-15 20:57:36 -04:00
|
|
|
const bodyArray = msg.bodyArray();
|
|
|
|
assert(bodyArray != null);
|
|
|
|
const ab = typedArrayToArrayBuffer(bodyArray!);
|
2018-07-06 11:20:35 -04:00
|
|
|
this.bodyWaiter.resolve(ab);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let nextFetchId = 0;
|
|
|
|
//TODO implements Request
|
|
|
|
class FetchRequest {
|
|
|
|
private readonly id: number;
|
|
|
|
response: FetchResponse;
|
|
|
|
constructor(readonly url: string) {
|
|
|
|
this.id = nextFetchId++;
|
|
|
|
fetchRequests.set(this.id, this);
|
|
|
|
this.response = new FetchResponse(this);
|
|
|
|
}
|
|
|
|
|
2018-08-15 20:57:36 -04:00
|
|
|
onMsg(base: fbs.Base, msg: fbs.FetchRes) {
|
|
|
|
this.response.onMsg(base, msg);
|
2018-07-06 11:20:35 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
destroy() {
|
|
|
|
fetchRequests.delete(this.id);
|
|
|
|
}
|
|
|
|
|
|
|
|
start() {
|
|
|
|
log("dispatch FETCH_REQ", this.id, this.url);
|
2018-08-15 20:57:36 -04:00
|
|
|
|
|
|
|
// Send FetchReq message
|
|
|
|
const builder = new flatbuffers.Builder();
|
|
|
|
const url = builder.createString(this.url);
|
|
|
|
fbs.FetchReq.startFetchReq(builder);
|
|
|
|
fbs.FetchReq.addId(builder, this.id);
|
|
|
|
fbs.FetchReq.addUrl(builder, url);
|
|
|
|
const msg = fbs.FetchReq.endFetchReq(builder);
|
|
|
|
fbs.Base.startBase(builder);
|
|
|
|
fbs.Base.addMsg(builder, msg);
|
|
|
|
fbs.Base.addMsgType(builder, fbs.Any.FetchReq);
|
|
|
|
builder.finish(fbs.Base.endBase(builder));
|
|
|
|
const resBuf = libdeno.send(builder.asUint8Array());
|
|
|
|
assert(resBuf == null);
|
|
|
|
|
|
|
|
//console.log("FetchReq sent", builder);
|
2018-07-06 11:20:35 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function fetch(
|
|
|
|
input?: Request | string,
|
|
|
|
init?: RequestInit
|
|
|
|
): Promise<Response> {
|
|
|
|
const fetchReq = new FetchRequest(input as string);
|
|
|
|
const response = fetchReq.response;
|
|
|
|
return new Promise((resolve, reject) => {
|
2018-08-15 20:57:36 -04:00
|
|
|
response.onHeader = (response: FetchResponse) => {
|
2018-07-06 11:20:35 -04:00
|
|
|
log("onHeader");
|
|
|
|
resolve(response);
|
|
|
|
};
|
|
|
|
response.onError = (error: Error) => {
|
|
|
|
log("onError", error);
|
|
|
|
reject(error);
|
|
|
|
};
|
|
|
|
fetchReq.start();
|
|
|
|
});
|
|
|
|
}
|