@appium/support 2.61.0 → 3.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/build/lib/console.d.ts +1 -1
- package/build/lib/console.js +169 -105
- package/build/lib/console.js.map +1 -1
- package/build/lib/env.js +142 -117
- package/build/lib/env.js.map +1 -1
- package/build/lib/fs.d.ts +9 -2
- package/build/lib/fs.d.ts.map +1 -1
- package/build/lib/fs.js +358 -246
- package/build/lib/fs.js.map +1 -1
- package/build/lib/image-util.js +139 -124
- package/build/lib/image-util.js.map +1 -1
- package/build/lib/index.js +64 -103
- package/build/lib/index.js.map +1 -1
- package/build/lib/log-internal.d.ts +4 -27
- package/build/lib/log-internal.d.ts.map +1 -1
- package/build/lib/log-internal.js +141 -123
- package/build/lib/log-internal.js.map +1 -1
- package/build/lib/logger.d.ts +1 -1
- package/build/lib/logger.d.ts.map +1 -1
- package/build/lib/logger.js +5 -14
- package/build/lib/logger.js.map +1 -1
- package/build/lib/logging.d.ts +3 -4
- package/build/lib/logging.d.ts.map +1 -1
- package/build/lib/logging.js +139 -110
- package/build/lib/logging.js.map +1 -1
- package/build/lib/mjpeg.js +169 -141
- package/build/lib/mjpeg.js.map +1 -1
- package/build/lib/mkdirp.js +7 -13
- package/build/lib/mkdirp.js.map +1 -1
- package/build/lib/net.d.ts.map +1 -1
- package/build/lib/net.js +278 -254
- package/build/lib/net.js.map +1 -1
- package/build/lib/node.js +203 -192
- package/build/lib/node.js.map +1 -1
- package/build/lib/npm.d.ts +19 -4
- package/build/lib/npm.d.ts.map +1 -1
- package/build/lib/npm.js +277 -228
- package/build/lib/npm.js.map +1 -1
- package/build/lib/plist.js +145 -136
- package/build/lib/plist.js.map +1 -1
- package/build/lib/process.js +41 -42
- package/build/lib/process.js.map +1 -1
- package/build/lib/system.js +39 -56
- package/build/lib/system.js.map +1 -1
- package/build/lib/tempdir.js +112 -73
- package/build/lib/tempdir.js.map +1 -1
- package/build/lib/timing.js +99 -84
- package/build/lib/timing.js.map +1 -1
- package/build/lib/util.js +454 -356
- package/build/lib/util.js.map +1 -1
- package/build/lib/zip.js +469 -423
- package/build/lib/zip.js.map +1 -1
- package/build/tsconfig.tsbuildinfo +1 -1
- package/lib/fs.js +15 -1
- package/lib/log-internal.js +12 -16
- package/lib/logging.js +2 -3
- package/lib/net.js +15 -6
- package/lib/npm.js +28 -18
- package/package.json +20 -19
package/build/lib/net.js
CHANGED
|
@@ -1,272 +1,296 @@
|
|
|
1
1
|
"use strict";
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
});
|
|
6
|
-
exports.downloadFile =
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
require("
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
var _util = require("./util");
|
|
18
|
-
|
|
19
|
-
var _logger = _interopRequireDefault(require("./logger"));
|
|
20
|
-
|
|
21
|
-
var _jsftp = _interopRequireDefault(require("jsftp"));
|
|
22
|
-
|
|
23
|
-
var _timing = _interopRequireDefault(require("./timing"));
|
|
24
|
-
|
|
25
|
-
var _axios = _interopRequireDefault(require("axios"));
|
|
26
|
-
|
|
27
|
-
var _formData = _interopRequireDefault(require("form-data"));
|
|
28
|
-
|
|
29
|
-
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
30
|
-
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.downloadFile = exports.uploadFile = void 0;
|
|
7
|
+
const lodash_1 = __importDefault(require("lodash"));
|
|
8
|
+
const fs_1 = __importDefault(require("./fs"));
|
|
9
|
+
const bluebird_1 = __importDefault(require("bluebird"));
|
|
10
|
+
const util_1 = require("./util");
|
|
11
|
+
const logger_1 = __importDefault(require("./logger"));
|
|
12
|
+
const jsftp_1 = __importDefault(require("jsftp"));
|
|
13
|
+
const timing_1 = __importDefault(require("./timing"));
|
|
14
|
+
const axios_1 = __importDefault(require("axios"));
|
|
15
|
+
const form_data_1 = __importDefault(require("form-data"));
|
|
31
16
|
const DEFAULT_TIMEOUT_MS = 4 * 60 * 1000;
|
|
32
|
-
|
|
17
|
+
/**
|
|
18
|
+
* Type guard for param to {@linkcode toAxiosAuth}
|
|
19
|
+
* @param {any} value
|
|
20
|
+
* @returns {value is AuthCredentials | import('axios').AxiosBasicCredentials}
|
|
21
|
+
*/
|
|
22
|
+
function isAxiosAuth(value) {
|
|
23
|
+
return lodash_1.default.isPlainObject(value);
|
|
24
|
+
}
|
|
25
|
+
/**
|
|
26
|
+
* Converts {@linkcode AuthCredentials} to credentials understood by {@linkcode axios}.
|
|
27
|
+
* @param {AuthCredentials | import('axios').AxiosBasicCredentials} [auth]
|
|
28
|
+
* @returns {import('axios').AxiosBasicCredentials?}
|
|
29
|
+
*/
|
|
33
30
|
function toAxiosAuth(auth) {
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
return axiosAuth.username && axiosAuth.password ? axiosAuth : null;
|
|
31
|
+
if (!isAxiosAuth(auth)) {
|
|
32
|
+
return null;
|
|
33
|
+
}
|
|
34
|
+
const axiosAuth = {
|
|
35
|
+
username: 'username' in auth ? auth.username : auth.user,
|
|
36
|
+
password: 'password' in auth ? auth.password : auth.pass,
|
|
37
|
+
};
|
|
38
|
+
return axiosAuth.username && axiosAuth.password ? axiosAuth : null;
|
|
43
39
|
}
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
auth,
|
|
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
|
-
pairs = _lodash.default.toPairs(formFields);
|
|
81
|
-
}
|
|
82
|
-
|
|
83
|
-
for (const [key, value] of pairs) {
|
|
84
|
-
if (_lodash.default.toLower(key) !== _lodash.default.toLower(fileFieldName)) {
|
|
85
|
-
form.append(key, value);
|
|
40
|
+
/**
|
|
41
|
+
* @param {NodeJS.ReadableStream} localFileStream
|
|
42
|
+
* @param {URL} parsedUri
|
|
43
|
+
* @param {HttpUploadOptions & NetOptions} [uploadOptions]
|
|
44
|
+
*/
|
|
45
|
+
async function uploadFileToHttp(localFileStream, parsedUri, uploadOptions = /** @type {HttpUploadOptions & NetOptions} */ ({})) {
|
|
46
|
+
const { method = 'POST', timeout = DEFAULT_TIMEOUT_MS, headers, auth, fileFieldName = 'file', formFields, } = uploadOptions;
|
|
47
|
+
const { href } = parsedUri;
|
|
48
|
+
/** @type {import('axios').AxiosRequestConfig} */
|
|
49
|
+
const requestOpts = {
|
|
50
|
+
url: href,
|
|
51
|
+
method,
|
|
52
|
+
timeout,
|
|
53
|
+
maxContentLength: Infinity,
|
|
54
|
+
maxBodyLength: Infinity,
|
|
55
|
+
};
|
|
56
|
+
const axiosAuth = toAxiosAuth(auth);
|
|
57
|
+
if (axiosAuth) {
|
|
58
|
+
requestOpts.auth = axiosAuth;
|
|
59
|
+
}
|
|
60
|
+
if (fileFieldName) {
|
|
61
|
+
const form = new form_data_1.default();
|
|
62
|
+
form.append(fileFieldName, localFileStream);
|
|
63
|
+
if (formFields) {
|
|
64
|
+
let pairs = [];
|
|
65
|
+
if (lodash_1.default.isArray(formFields)) {
|
|
66
|
+
pairs = formFields;
|
|
67
|
+
}
|
|
68
|
+
else if (lodash_1.default.isPlainObject(formFields)) {
|
|
69
|
+
pairs = lodash_1.default.toPairs(formFields);
|
|
70
|
+
}
|
|
71
|
+
for (const [key, value] of pairs) {
|
|
72
|
+
if (lodash_1.default.toLower(key) !== lodash_1.default.toLower(fileFieldName)) {
|
|
73
|
+
form.append(key, value);
|
|
74
|
+
}
|
|
75
|
+
}
|
|
86
76
|
}
|
|
87
|
-
|
|
77
|
+
requestOpts.headers = {
|
|
78
|
+
...(lodash_1.default.isPlainObject(headers) ? headers : {}),
|
|
79
|
+
...form.getHeaders(),
|
|
80
|
+
};
|
|
81
|
+
requestOpts.data = form;
|
|
88
82
|
}
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
} else {
|
|
95
|
-
if (_lodash.default.isPlainObject(headers)) {
|
|
96
|
-
requestOpts.headers = headers;
|
|
83
|
+
else {
|
|
84
|
+
if (lodash_1.default.isPlainObject(headers)) {
|
|
85
|
+
requestOpts.headers = headers;
|
|
86
|
+
}
|
|
87
|
+
requestOpts.data = localFileStream;
|
|
97
88
|
}
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
_logger.default.debug(`Performing ${method} to ${href} with options (excluding data): ` + JSON.stringify(_lodash.default.omit(requestOpts, ['data'])));
|
|
103
|
-
|
|
104
|
-
const {
|
|
105
|
-
status,
|
|
106
|
-
statusText
|
|
107
|
-
} = await (0, _axios.default)(requestOpts);
|
|
108
|
-
|
|
109
|
-
_logger.default.info(`Server response: ${status} ${statusText}`);
|
|
89
|
+
logger_1.default.debug(`Performing ${method} to ${href} with options (excluding data): ` +
|
|
90
|
+
JSON.stringify(lodash_1.default.omit(requestOpts, ['data'])));
|
|
91
|
+
const { status, statusText } = await (0, axios_1.default)(requestOpts);
|
|
92
|
+
logger_1.default.info(`Server response: ${status} ${statusText}`);
|
|
110
93
|
}
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
port,
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
reject(err);
|
|
138
|
-
} else {
|
|
139
|
-
resolve();
|
|
140
|
-
}
|
|
94
|
+
/**
|
|
95
|
+
* @param {string | Buffer | NodeJS.ReadableStream} localFileStream
|
|
96
|
+
* @param {URL} parsedUri
|
|
97
|
+
* @param {NotHttpUploadOptions & NetOptions} [uploadOptions]
|
|
98
|
+
*/
|
|
99
|
+
async function uploadFileToFtp(localFileStream, parsedUri, uploadOptions = /** @type {NotHttpUploadOptions & NetOptions} */ ({})) {
|
|
100
|
+
const { auth } = uploadOptions;
|
|
101
|
+
const { hostname, port, protocol, pathname } = parsedUri;
|
|
102
|
+
const ftpOpts = {
|
|
103
|
+
host: hostname,
|
|
104
|
+
port: !lodash_1.default.isUndefined(port) ? lodash_1.default.parseInt(port) : 21,
|
|
105
|
+
};
|
|
106
|
+
if (auth?.user && auth?.pass) {
|
|
107
|
+
ftpOpts.user = auth.user;
|
|
108
|
+
ftpOpts.pass = auth.pass;
|
|
109
|
+
}
|
|
110
|
+
logger_1.default.debug(`${protocol} upload options: ${JSON.stringify(ftpOpts)}`);
|
|
111
|
+
return await new bluebird_1.default((resolve, reject) => {
|
|
112
|
+
new jsftp_1.default(ftpOpts).put(localFileStream, pathname, (err) => {
|
|
113
|
+
if (err) {
|
|
114
|
+
reject(err);
|
|
115
|
+
}
|
|
116
|
+
else {
|
|
117
|
+
resolve();
|
|
118
|
+
}
|
|
119
|
+
});
|
|
141
120
|
});
|
|
142
|
-
});
|
|
143
121
|
}
|
|
144
|
-
|
|
122
|
+
/**
|
|
123
|
+
* Returns `true` if params are valid for {@linkcode uploadFileToHttp}.
|
|
124
|
+
* @param {any} opts
|
|
125
|
+
* @param {URL} url
|
|
126
|
+
* @returns {opts is HttpUploadOptions & NetOptions}
|
|
127
|
+
*/
|
|
145
128
|
function isHttpUploadOptions(opts, url) {
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
}
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
}
|
|
129
|
+
try {
|
|
130
|
+
const { protocol } = url;
|
|
131
|
+
return protocol === 'http:' || protocol === 'https:';
|
|
132
|
+
}
|
|
133
|
+
catch {
|
|
134
|
+
return false;
|
|
135
|
+
}
|
|
154
136
|
}
|
|
155
|
-
|
|
137
|
+
/**
|
|
138
|
+
* Returns `true` if params are valid for {@linkcode uploadFileToFtp}.
|
|
139
|
+
* @param {any} opts
|
|
140
|
+
* @param {URL} url
|
|
141
|
+
* @returns {opts is NotHttpUploadOptions & NetOptions}
|
|
142
|
+
*/
|
|
156
143
|
function isNotHttpUploadOptions(opts, url) {
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
}
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
}
|
|
144
|
+
try {
|
|
145
|
+
const { protocol } = url;
|
|
146
|
+
return protocol === 'ftp:';
|
|
147
|
+
}
|
|
148
|
+
catch {
|
|
149
|
+
return false;
|
|
150
|
+
}
|
|
165
151
|
}
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
152
|
+
/**
|
|
153
|
+
* Uploads the given file to a remote location. HTTP(S) and FTP
|
|
154
|
+
* protocols are supported.
|
|
155
|
+
*
|
|
156
|
+
* @param {string} localPath - The path to a file on the local storage.
|
|
157
|
+
* @param {string} remoteUri - The remote URI to upload the file to.
|
|
158
|
+
* @param {(HttpUploadOptions|NotHttpUploadOptions) & NetOptions} [uploadOptions]
|
|
159
|
+
* @returns {Promise<void>}
|
|
160
|
+
*/
|
|
161
|
+
async function uploadFile(localPath, remoteUri, uploadOptions = /** @type {(HttpUploadOptions|NotHttpUploadOptions) & NetOptions} */ ({})) {
|
|
162
|
+
if (!(await fs_1.default.exists(localPath))) {
|
|
163
|
+
throw new Error(`'${localPath}' does not exists or is not accessible`);
|
|
164
|
+
}
|
|
165
|
+
const { isMetered = true } = uploadOptions;
|
|
166
|
+
const url = new URL(remoteUri);
|
|
167
|
+
const { size } = await fs_1.default.stat(localPath);
|
|
168
|
+
if (isMetered) {
|
|
169
|
+
logger_1.default.info(`Uploading '${localPath}' of ${(0, util_1.toReadableSizeString)(size)} size to '${remoteUri}'`);
|
|
170
|
+
}
|
|
171
|
+
const timer = new timing_1.default().start();
|
|
172
|
+
if (isHttpUploadOptions(uploadOptions, url)) {
|
|
173
|
+
if (!uploadOptions.fileFieldName) {
|
|
174
|
+
uploadOptions.headers = {
|
|
175
|
+
...(lodash_1.default.isPlainObject(uploadOptions.headers) ? uploadOptions.headers : {}),
|
|
176
|
+
'Content-Length': size,
|
|
177
|
+
};
|
|
178
|
+
}
|
|
179
|
+
await uploadFileToHttp(fs_1.default.createReadStream(localPath), url, uploadOptions);
|
|
180
|
+
}
|
|
181
|
+
else if (isNotHttpUploadOptions(uploadOptions, url)) {
|
|
182
|
+
await uploadFileToFtp(fs_1.default.createReadStream(localPath), url, uploadOptions);
|
|
183
|
+
}
|
|
184
|
+
else {
|
|
185
|
+
throw new Error(`Cannot upload the file at '${localPath}' to '${remoteUri}'. ` +
|
|
186
|
+
`Unsupported remote protocol '${url.protocol}'. ` +
|
|
187
|
+
`Only http/https and ftp/ftps protocols are supported.`);
|
|
188
|
+
}
|
|
189
|
+
if (isMetered) {
|
|
190
|
+
logger_1.default.info(`Uploaded '${localPath}' of ${(0, util_1.toReadableSizeString)(size)} size in ` +
|
|
191
|
+
`${timer.getDuration().asSeconds.toFixed(3)}s`);
|
|
191
192
|
}
|
|
192
|
-
|
|
193
|
-
await uploadFileToHttp(_fs.default.createReadStream(localPath), url, uploadOptions);
|
|
194
|
-
} else if (isNotHttpUploadOptions(uploadOptions, url)) {
|
|
195
|
-
await uploadFileToFtp(_fs.default.createReadStream(localPath), url, uploadOptions);
|
|
196
|
-
} else {
|
|
197
|
-
throw new Error(`Cannot upload the file at '${localPath}' to '${remoteUri}'. ` + `Unsupported remote protocol '${url.protocol}'. ` + `Only http/https and ftp/ftps protocols are supported.`);
|
|
198
|
-
}
|
|
199
|
-
|
|
200
|
-
if (isMetered) {
|
|
201
|
-
_logger.default.info(`Uploaded '${localPath}' of ${(0, _util.toReadableSizeString)(size)} size in ` + `${timer.getDuration().asSeconds.toFixed(3)}s`);
|
|
202
|
-
}
|
|
203
193
|
}
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
_logger.default.debug(`${remoteUrl} (${(0, _util.toReadableSizeString)(size)}) ` + `has been downloaded to '${dstPath}' in ${secondsElapsed.toFixed(3)}s`);
|
|
264
|
-
|
|
265
|
-
if (secondsElapsed >= 2) {
|
|
266
|
-
const bytesPerSec = Math.floor(size / secondsElapsed);
|
|
267
|
-
|
|
268
|
-
_logger.default.debug(`Approximate download speed: ${(0, _util.toReadableSizeString)(bytesPerSec)}/s`);
|
|
194
|
+
exports.uploadFile = uploadFile;
|
|
195
|
+
/**
|
|
196
|
+
* Downloads the given file via HTTP(S)
|
|
197
|
+
*
|
|
198
|
+
* @param {string} remoteUrl - The remote url
|
|
199
|
+
* @param {string} dstPath - The local path to download the file to
|
|
200
|
+
* @param {DownloadOptions & NetOptions} [downloadOptions]
|
|
201
|
+
* @throws {Error} If download operation fails
|
|
202
|
+
*/
|
|
203
|
+
async function downloadFile(remoteUrl, dstPath, downloadOptions = /** @type {DownloadOptions & NetOptions} */ ({})) {
|
|
204
|
+
const { isMetered = true, auth, timeout = DEFAULT_TIMEOUT_MS, headers } = downloadOptions;
|
|
205
|
+
/**
|
|
206
|
+
* @type {import('axios').AxiosRequestConfig}
|
|
207
|
+
*/
|
|
208
|
+
const requestOpts = {
|
|
209
|
+
url: remoteUrl,
|
|
210
|
+
responseType: 'stream',
|
|
211
|
+
timeout,
|
|
212
|
+
};
|
|
213
|
+
const axiosAuth = toAxiosAuth(auth);
|
|
214
|
+
if (axiosAuth) {
|
|
215
|
+
requestOpts.auth = axiosAuth;
|
|
216
|
+
}
|
|
217
|
+
if (lodash_1.default.isPlainObject(headers)) {
|
|
218
|
+
requestOpts.headers = headers;
|
|
219
|
+
}
|
|
220
|
+
const timer = new timing_1.default().start();
|
|
221
|
+
let responseLength;
|
|
222
|
+
try {
|
|
223
|
+
const writer = fs_1.default.createWriteStream(dstPath);
|
|
224
|
+
const { data: responseStream, headers: responseHeaders } = await (0, axios_1.default)(requestOpts);
|
|
225
|
+
responseLength = parseInt(responseHeaders['content-length'] || '0', 10);
|
|
226
|
+
responseStream.pipe(writer);
|
|
227
|
+
await new bluebird_1.default((resolve, reject) => {
|
|
228
|
+
responseStream.once('error', reject);
|
|
229
|
+
writer.once('finish', resolve);
|
|
230
|
+
writer.once('error', (e) => {
|
|
231
|
+
responseStream.unpipe(writer);
|
|
232
|
+
reject(e);
|
|
233
|
+
});
|
|
234
|
+
});
|
|
235
|
+
}
|
|
236
|
+
catch (err) {
|
|
237
|
+
throw new Error(`Cannot download the file from ${remoteUrl}: ${err.message}`);
|
|
238
|
+
}
|
|
239
|
+
const { size } = await fs_1.default.stat(dstPath);
|
|
240
|
+
if (responseLength && size !== responseLength) {
|
|
241
|
+
await fs_1.default.rimraf(dstPath);
|
|
242
|
+
throw new Error(`The size of the file downloaded from ${remoteUrl} (${size} bytes) ` +
|
|
243
|
+
`differs from the one in Content-Length response header (${responseLength} bytes)`);
|
|
244
|
+
}
|
|
245
|
+
if (isMetered) {
|
|
246
|
+
const secondsElapsed = timer.getDuration().asSeconds;
|
|
247
|
+
logger_1.default.debug(`${remoteUrl} (${(0, util_1.toReadableSizeString)(size)}) ` +
|
|
248
|
+
`has been downloaded to '${dstPath}' in ${secondsElapsed.toFixed(3)}s`);
|
|
249
|
+
if (secondsElapsed >= 2) {
|
|
250
|
+
const bytesPerSec = Math.floor(size / secondsElapsed);
|
|
251
|
+
logger_1.default.debug(`Approximate download speed: ${(0, util_1.toReadableSizeString)(bytesPerSec)}/s`);
|
|
252
|
+
}
|
|
269
253
|
}
|
|
270
|
-
}
|
|
271
254
|
}
|
|
272
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["DEFAULT_TIMEOUT_MS","toAxiosAuth","auth","_","isPlainObject","axiosAuth","username","get","password","uploadFileToHttp","localFileStream","parsedUri","uploadOptions","method","timeout","headers","fileFieldName","formFields","href","requestOpts","url","maxContentLength","Infinity","maxBodyLength","form","FormData","append","pairs","isArray","toPairs","key","value","toLower","getHeaders","data","log","debug","JSON","stringify","omit","status","statusText","axios","info","uploadFileToFtp","hostname","port","protocol","pathname","ftpOpts","host","isUndefined","parseInt","user","pass","B","resolve","reject","Ftp","put","err","isHttpUploadOptions","opts","URL","isNotHttpUploadOptions","uploadFile","localPath","remoteUri","fs","exists","Error","isMetered","size","stat","toReadableSizeString","timer","Timer","start","createReadStream","getDuration","asSeconds","toFixed","downloadFile","remoteUrl","dstPath","downloadOptions","responseType","responseLength","writer","createWriteStream","responseStream","responseHeaders","pipe","once","e","unpipe","message","rimraf","secondsElapsed","bytesPerSec","Math","floor"],"sources":["../../lib/net.js"],"sourcesContent":["import _ from 'lodash';\nimport fs from './fs';\nimport B from 'bluebird';\nimport {toReadableSizeString} from './util';\nimport log from './logger';\nimport Ftp from 'jsftp';\nimport Timer from './timing';\nimport axios from 'axios';\nimport FormData from 'form-data';\n\nconst DEFAULT_TIMEOUT_MS = 4 * 60 * 1000;\n\n/**\n * Converts {@linkcode AuthCredentials} to credentials understood by {@linkcode axios}.\n * @param {AuthCredentials | import('axios').AxiosBasicCredentials} [auth]\n * @returns {import('axios').AxiosBasicCredentials?}\n */\nfunction toAxiosAuth(auth) {\n  if (!_.isPlainObject(auth)) {\n    return null;\n  }\n\n  const axiosAuth = {\n    username: _.get(auth, 'username', _.get(auth, 'user')),\n    password: _.get(auth, 'password', _.get(auth, 'pass')),\n  };\n  return axiosAuth.username && axiosAuth.password ? axiosAuth : null;\n}\n\n/**\n * @param {NodeJS.ReadableStream} localFileStream\n * @param {URL} parsedUri\n * @param {HttpUploadOptions & NetOptions} [uploadOptions]\n */\nasync function uploadFileToHttp(\n  localFileStream,\n  parsedUri,\n  uploadOptions = /** @type {HttpUploadOptions & NetOptions} */ ({})\n) {\n  const {\n    method = 'POST',\n    timeout = DEFAULT_TIMEOUT_MS,\n    headers,\n    auth,\n    fileFieldName = 'file',\n    formFields,\n  } = uploadOptions;\n  const {href} = parsedUri;\n\n  /** @type {import('axios').AxiosRequestConfig} */\n  const requestOpts = {\n    url: href,\n    method,\n    timeout,\n    maxContentLength: Infinity,\n    maxBodyLength: Infinity,\n  };\n  const axiosAuth = toAxiosAuth(auth);\n  if (axiosAuth) {\n    requestOpts.auth = axiosAuth;\n  }\n  if (fileFieldName) {\n    const form = new FormData();\n    form.append(fileFieldName, localFileStream);\n    if (formFields) {\n      let pairs = [];\n      if (_.isArray(formFields)) {\n        pairs = formFields;\n      } else if (_.isPlainObject(formFields)) {\n        pairs = _.toPairs(formFields);\n      }\n      for (const [key, value] of pairs) {\n        if (_.toLower(key) !== _.toLower(fileFieldName)) {\n          form.append(key, value);\n        }\n      }\n    }\n    requestOpts.headers = {\n      ...(_.isPlainObject(headers) ? headers : {}),\n      ...form.getHeaders(),\n    };\n    requestOpts.data = form;\n  } else {\n    if (_.isPlainObject(headers)) {\n      requestOpts.headers = headers;\n    }\n    requestOpts.data = localFileStream;\n  }\n  log.debug(\n    `Performing ${method} to ${href} with options (excluding data): ` +\n      JSON.stringify(_.omit(requestOpts, ['data']))\n  );\n\n  const {status, statusText} = await axios(requestOpts);\n  log.info(`Server response: ${status} ${statusText}`);\n}\n\n/**\n * @param {string | Buffer | NodeJS.ReadableStream} localFileStream\n * @param {URL} parsedUri\n * @param {NotHttpUploadOptions & NetOptions} [uploadOptions]\n */\nasync function uploadFileToFtp(\n  localFileStream,\n  parsedUri,\n  uploadOptions = /** @type {NotHttpUploadOptions & NetOptions} */ ({})\n) {\n  const {auth} = uploadOptions;\n  const {hostname, port, protocol, pathname} = parsedUri;\n\n  const ftpOpts = {\n    host: hostname,\n    port: !_.isUndefined(port) ? _.parseInt(port) : 21,\n  };\n  if (auth?.user && auth?.pass) {\n    ftpOpts.user = auth.user;\n    ftpOpts.pass = auth.pass;\n  }\n  log.debug(`${protocol} upload options: ${JSON.stringify(ftpOpts)}`);\n  return await new B((resolve, reject) => {\n    new Ftp(ftpOpts).put(localFileStream, pathname, (err) => {\n      if (err) {\n        reject(err);\n      } else {\n        resolve();\n      }\n    });\n  });\n}\n\n/**\n * Returns `true` if params are valid for {@linkcode uploadFileToHttp}.\n * @param {any} opts\n * @param {URL} url\n * @returns {opts is HttpUploadOptions & NetOptions}\n */\nfunction isHttpUploadOptions(opts, url) {\n  try {\n    const {protocol} = new URL(url);\n    return protocol === 'http:' || protocol === 'https:';\n  } catch {\n    return false;\n  }\n}\n\n/**\n * Returns `true` if params are valid for {@linkcode uploadFileToFtp}.\n * @param {any} opts\n * @param {URL} url\n * @returns {opts is NotHttpUploadOptions & NetOptions}\n */\nfunction isNotHttpUploadOptions(opts, url) {\n  try {\n    const {protocol} = new URL(url);\n    return protocol === 'ftp:';\n  } catch {\n    return false;\n  }\n}\n/**\n * Uploads the given file to a remote location. HTTP(S) and FTP\n * protocols are supported.\n *\n * @param {string} localPath - The path to a file on the local storage.\n * @param {string} remoteUri - The remote URI to upload the file to.\n * @param {(HttpUploadOptions|NotHttpUploadOptions) & NetOptions} [uploadOptions]\n * @returns {Promise<void>}\n */\nasync function uploadFile(\n  localPath,\n  remoteUri,\n  uploadOptions = /** @type {(HttpUploadOptions|NotHttpUploadOptions) & NetOptions} */ ({})\n) {\n  if (!(await fs.exists(localPath))) {\n    throw new Error(`'${localPath}' does not exists or is not accessible`);\n  }\n\n  const {isMetered = true} = uploadOptions;\n  const url = new URL(remoteUri);\n  const {size} = await fs.stat(localPath);\n  if (isMetered) {\n    log.info(`Uploading '${localPath}' of ${toReadableSizeString(size)} size to '${remoteUri}'`);\n  }\n  const timer = new Timer().start();\n  if (isHttpUploadOptions(uploadOptions, url)) {\n    if (!uploadOptions.fileFieldName) {\n      uploadOptions.headers = {\n        ...(_.isPlainObject(uploadOptions.headers) ? uploadOptions.headers : {}),\n        'Content-Length': size,\n      };\n    }\n    await uploadFileToHttp(fs.createReadStream(localPath), url, uploadOptions);\n  } else if (isNotHttpUploadOptions(uploadOptions, url)) {\n    await uploadFileToFtp(fs.createReadStream(localPath), url, uploadOptions);\n  } else {\n    throw new Error(\n      `Cannot upload the file at '${localPath}' to '${remoteUri}'. ` +\n        `Unsupported remote protocol '${url.protocol}'. ` +\n        `Only http/https and ftp/ftps protocols are supported.`\n    );\n  }\n  if (isMetered) {\n    log.info(\n      `Uploaded '${localPath}' of ${toReadableSizeString(size)} size in ` +\n        `${timer.getDuration().asSeconds.toFixed(3)}s`\n    );\n  }\n}\n\n/**\n * Downloads the given file via HTTP(S)\n *\n * @param {string} remoteUrl - The remote url\n * @param {string} dstPath - The local path to download the file to\n * @param {DownloadOptions & NetOptions} [downloadOptions]\n * @throws {Error} If download operation fails\n */\nasync function downloadFile(\n  remoteUrl,\n  dstPath,\n  downloadOptions = /** @type {DownloadOptions & NetOptions} */ ({})\n) {\n  const {isMetered = true, auth, timeout = DEFAULT_TIMEOUT_MS, headers} = downloadOptions;\n\n  /**\n   * @type {import('axios').AxiosRequestConfig}\n   */\n  const requestOpts = {\n    url: remoteUrl,\n    responseType: 'stream',\n    timeout,\n  };\n  const axiosAuth = toAxiosAuth(auth);\n  if (axiosAuth) {\n    requestOpts.auth = axiosAuth;\n  }\n  if (_.isPlainObject(headers)) {\n    requestOpts.headers = headers;\n  }\n\n  const timer = new Timer().start();\n  let responseLength;\n  try {\n    const writer = fs.createWriteStream(dstPath);\n    const {data: responseStream, headers: responseHeaders} = await axios(requestOpts);\n    responseLength = parseInt(responseHeaders['content-length'], 10);\n    responseStream.pipe(writer);\n\n    await new B((resolve, reject) => {\n      responseStream.once('error', reject);\n      writer.once('finish', resolve);\n      writer.once('error', (e) => {\n        responseStream.unpipe(writer);\n        reject(e);\n      });\n    });\n  } catch (err) {\n    throw new Error(`Cannot download the file from ${remoteUrl}: ${err.message}`);\n  }\n\n  const {size} = await fs.stat(dstPath);\n  if (responseLength && size !== responseLength) {\n    await fs.rimraf(dstPath);\n    throw new Error(\n      `The size of the file downloaded from ${remoteUrl} (${size} bytes) ` +\n        `differs from the one in Content-Length response header (${responseLength} bytes)`\n    );\n  }\n  if (isMetered) {\n    const secondsElapsed = timer.getDuration().asSeconds;\n    log.debug(\n      `${remoteUrl} (${toReadableSizeString(size)}) ` +\n        `has been downloaded to '${dstPath}' in ${secondsElapsed.toFixed(3)}s`\n    );\n    if (secondsElapsed >= 2) {\n      const bytesPerSec = Math.floor(size / secondsElapsed);\n      log.debug(`Approximate download speed: ${toReadableSizeString(bytesPerSec)}/s`);\n    }\n  }\n}\n\nexport {uploadFile, downloadFile};\n\n/**\n * Common options for {@linkcode uploadFile} and {@linkcode downloadFile}.\n * @typedef NetOptions\n * @property {boolean} [isMetered=true] - Whether to log the actual download performance\n * (e.g. timings and speed)\n * @property {AuthCredentials} [auth] - Authentication credentials\n */\n\n/**\n * Specific options for {@linkcode downloadFile}.\n * @typedef DownloadOptions\n * @property {number} [timeout] - The actual request timeout in milliseconds; defaults to {@linkcode DEFAULT_TIMEOUT_MS}\n * @property {Record<string,any>} [headers] - Request headers mapping\n */\n\n/**\n * Basic auth credentials; used by {@linkcode NetOptions}.\n * @typedef AuthCredentials\n * @property {string} user - Non-empty user name\n * @property {string} pass - Non-empty password\n */\n\n/**\n * This type is used in {@linkcode uploadFile} if the remote location uses the `ftp` protocol, and distinguishes the type from {@linkcode HttpUploadOptions}.\n * @typedef NotHttpUploadOptions\n * @property {never} headers\n * @property {never} method\n * @property {never} timeout\n * @property {never} fileFieldName\n * @property {never} formFields\n */\n\n/**\n * Specific options for {@linkcode uploadFile} if the remote location uses the `http(s)` protocol\n * @typedef HttpUploadOptions\n * @property {Record<string,any>} headers - Additional request headers mapping\n * @property {import('axios').Method} [method='POST'] - The HTTP method used for file upload\n * @property {number} [timeout] - The actual request timeout in milliseconds; defaults to {@linkcode DEFAULT_TIMEOUT_MS}\n * @property {string} [fileFieldName='file'] - The name of the form field containing the file\n * content to be uploaded. Any falsy value make the request to use non-multipart upload\n * @property {Record<string,any>} [formFields] - The additional form fields\n * to be included into the upload request. This property is only considered if\n * `fileFieldName` is set\n */\n"],"mappings":"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"}
|
|
255
|
+
exports.downloadFile = downloadFile;
|
|
256
|
+
/**
|
|
257
|
+
* Common options for {@linkcode uploadFile} and {@linkcode downloadFile}.
|
|
258
|
+
* @typedef NetOptions
|
|
259
|
+
* @property {boolean} [isMetered=true] - Whether to log the actual download performance
|
|
260
|
+
* (e.g. timings and speed)
|
|
261
|
+
* @property {AuthCredentials} [auth] - Authentication credentials
|
|
262
|
+
*/
|
|
263
|
+
/**
|
|
264
|
+
* Specific options for {@linkcode downloadFile}.
|
|
265
|
+
* @typedef DownloadOptions
|
|
266
|
+
* @property {number} [timeout] - The actual request timeout in milliseconds; defaults to {@linkcode DEFAULT_TIMEOUT_MS}
|
|
267
|
+
* @property {Record<string,any>} [headers] - Request headers mapping
|
|
268
|
+
*/
|
|
269
|
+
/**
|
|
270
|
+
* Basic auth credentials; used by {@linkcode NetOptions}.
|
|
271
|
+
* @typedef AuthCredentials
|
|
272
|
+
* @property {string} user - Non-empty user name
|
|
273
|
+
* @property {string} pass - Non-empty password
|
|
274
|
+
*/
|
|
275
|
+
/**
|
|
276
|
+
* This type is used in {@linkcode uploadFile} if the remote location uses the `ftp` protocol, and distinguishes the type from {@linkcode HttpUploadOptions}.
|
|
277
|
+
* @typedef NotHttpUploadOptions
|
|
278
|
+
* @property {never} headers
|
|
279
|
+
* @property {never} method
|
|
280
|
+
* @property {never} timeout
|
|
281
|
+
* @property {never} fileFieldName
|
|
282
|
+
* @property {never} formFields
|
|
283
|
+
*/
|
|
284
|
+
/**
|
|
285
|
+
* Specific options for {@linkcode uploadFile} if the remote location uses the `http(s)` protocol
|
|
286
|
+
* @typedef HttpUploadOptions
|
|
287
|
+
* @property {Record<string,any>} headers - Additional request headers mapping
|
|
288
|
+
* @property {import('axios').Method} [method='POST'] - The HTTP method used for file upload
|
|
289
|
+
* @property {number} [timeout] - The actual request timeout in milliseconds; defaults to {@linkcode DEFAULT_TIMEOUT_MS}
|
|
290
|
+
* @property {string} [fileFieldName='file'] - The name of the form field containing the file
|
|
291
|
+
* content to be uploaded. Any falsy value make the request to use non-multipart upload
|
|
292
|
+
* @property {Record<string,any>} [formFields] - The additional form fields
|
|
293
|
+
* to be included into the upload request. This property is only considered if
|
|
294
|
+
* `fileFieldName` is set
|
|
295
|
+
*/
|
|
296
|
+
//# sourceMappingURL=net.js.map
|