/
Request.ts
159 lines (148 loc) · 3.75 KB
/
Request.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
import * as NodeFetch from 'node-fetch';
import IRequest from './IRequest';
import IBlob from '../file/IBlob';
import IDocument from '../nodes/document/IDocument';
/**
* Fetch request.
*/
export default class Request extends NodeFetch.Request implements IRequest {
// Owner document is set by a sub-class in the Window constructor
public static _ownerDocument: IDocument = null;
public readonly _ownerDocument: IDocument = null;
/**
* Constructor.
*
* @param input Input.
* @param [init] Init.
*/
constructor(input: NodeFetch.RequestInfo, init?: NodeFetch.RequestInit) {
super(input, init);
this._ownerDocument = (<typeof Request>this.constructor)._ownerDocument;
}
/**
* Returns array buffer.
*
* @returns Array buffer.
*/
public arrayBuffer(): Promise<ArrayBuffer> {
return new Promise((resolve, reject) => {
const taskID = this._handlePromiseStart();
super
.arrayBuffer()
.then(this._handlePromiseEnd.bind(this, resolve, reject, taskID))
.catch(this._handlePromiseError.bind(this, reject));
});
}
/**
* Returns blob.
*
* @returns Blob.
*/
public blob(): Promise<IBlob> {
return new Promise((resolve, reject) => {
const taskID = this._handlePromiseStart();
super
.blob()
.then(this._handlePromiseEnd.bind(this, resolve, reject, taskID))
.catch(this._handlePromiseError.bind(this, reject));
});
}
/**
* Returns buffer.
*
* @returns Buffer.
*/
public buffer(): Promise<Buffer> {
return new Promise((resolve, reject) => {
const taskID = this._handlePromiseStart();
super
.buffer()
.then(this._handlePromiseEnd.bind(this, resolve, reject, taskID))
.catch(this._handlePromiseError.bind(this, reject));
});
}
/**
* Returns json.
*
* @returns JSON.
*/
public json(): Promise<unknown> {
return new Promise((resolve, reject) => {
const taskID = this._handlePromiseStart();
super
.json()
.then(this._handlePromiseEnd.bind(this, resolve, reject, taskID))
.catch(this._handlePromiseError.bind(this, reject));
});
}
/**
* Returns json.
*
* @returns JSON.
*/
public text(): Promise<string> {
return new Promise((resolve, reject) => {
const taskID = this._handlePromiseStart();
super
.text()
.then(this._handlePromiseEnd.bind(this, resolve, reject, taskID))
.catch(this._handlePromiseError.bind(this, reject));
});
}
/**
* Returns json.
*
* @returns JSON.
*/
public textConverted(): Promise<string> {
return new Promise((resolve, reject) => {
const taskID = this._handlePromiseStart();
super
.textConverted()
.then(this._handlePromiseEnd.bind(this, resolve, reject, taskID))
.catch(this._handlePromiseError.bind(this, reject));
});
}
/**
* Handles promise start.
*
* @returns Task ID.
*/
private _handlePromiseStart(): number {
const taskManager = this._ownerDocument.defaultView.happyDOM.asyncTaskManager;
return taskManager.startTask();
}
/**
* Handles promise end.
*
* @param resolve Resolve.
* @param reject Reject.
* @param taskID Task ID.
* @param response Response.
*/
private _handlePromiseEnd(
resolve: (response: unknown) => void,
reject: (error: Error) => void,
taskID: number,
response: unknown
): void {
const taskManager = this._ownerDocument.defaultView.happyDOM.asyncTaskManager;
if (taskManager.getTaskCount() === 0) {
reject(new Error('Failed to complete fetch request. Task was canceled.'));
} else {
resolve(response);
taskManager.endTask(taskID);
}
}
/**
* Handles promise error.
*
* @param error
* @param reject
*/
private _handlePromiseError(reject: (error: Error) => void, error: Error): void {
const taskManager = this._ownerDocument.defaultView.happyDOM.asyncTaskManager;
reject(error);
taskManager.cancelAll(error);
}
}