-
Notifications
You must be signed in to change notification settings - Fork 9
Expand file tree
/
Copy pathmsdo-nuget-client.ts
More file actions
783 lines (672 loc) · 27.6 KB
/
msdo-nuget-client.ts
File metadata and controls
783 lines (672 loc) · 27.6 KB
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
160
161
162
163
164
165
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
191
192
193
194
195
196
197
198
199
200
201
202
203
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
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
import * as https from 'https';
import { HttpsProxyAgent } from 'https-proxy-agent';
import * as fs from 'fs';
import * as path from 'path';
import * as process from 'process';
import * as tl from 'azure-pipelines-task-lib/task';
import AdmZip = require('adm-zip');
import * as common from './msdo-common';
/**
* The default number of times to retry downloading a file.
*/
const _defaultFileDownloadRetries = 2;
/**
* The default delay in milliseconds between file download retries.
*/
const _defaultFileDownloadRetryDelayMs = 1000;
/**
* Information about an installed nuget package
*/
export interface InstallNuGetPackageResponse {
success: boolean,
inCache?: boolean,
packageName: string,
packageVersion: string,
resolvedVersion?: string,
packageFolder?: string,
packagePath?: string
}
/**
* The response type when finding a service from the Service Index
* This format is used by callService() to attempt to use all service endpoints,
* known and unknown, to ensure service resiliency
*/
export interface FindServiceResponse {
known: NuGetServiceResource[],
unknown: NuGetServiceResource[]
}
/**
* An extended format of the Service Index's resource object,
* splitting out the name and version from @type for easier processing
*/
export interface NuGetServiceResource {
'@id': string,
'@type': string,
'name': string,
'version': string
}
/**
* Installs a NuGet package from a NuGet server starting at the service index url.
*
* @param serviceIndexUrl - The url to the service index of the NuGet server to install from
* @param packageName - The name of the package to install
* @param packageVersion - The version of the package to install, accepts exact values or "Latest" and "LatestPreRelease" respectively.
* @param outputDirectory - The directory to install the package to
* @param accessToken - (optional) The access token to use when authenticating to the NuGet server
* @returns An object with information about the installed package.
*/
export async function install(
serviceIndexUrl: string,
packageName: string,
packageVersion: string,
outputDirectory: string,
accessToken: string = null): Promise<InstallNuGetPackageResponse> {
let response = await getInstallationStatus(packageName, packageVersion, outputDirectory);
if (response.inCache) {
tl.debug(`Package already installed: ${packageName} ${packageVersion}`);
} else {
let requestOptions = resolveRequestOptions(accessToken);
tl.debug(`Fetching service index for: ${serviceIndexUrl}`);
let serviceIndex = await requestJson(serviceIndexUrl, requestOptions);
let resolvedVersion = packageVersion;
if (common.isLatest(packageVersion)) {
tl.debug(`Resolving package name and version: ${packageName} ${packageVersion}`);
resolvedVersion = await resolveVersion(serviceIndex, requestOptions, packageName, packageVersion);
response = await getInstallationStatus(packageName, resolvedVersion, outputDirectory, true);
}
if (response.inCache) {
tl.debug(`Resolved package already installed: ${packageName} ${resolvedVersion}`);
} else {
tl.debug(`Downloading package to: ${outputDirectory}`);
let packagePath = await downloadPackage(serviceIndex, requestOptions, packageName, resolvedVersion, outputDirectory);
// extract the package
tl.debug(`Extracting package: ${packagePath}`);
await extractPackage(packagePath);
response['success'] = true;
response['resolvedVersion'] = resolvedVersion;
response['packageFolder'] = common.removeExtension(packagePath);
response['packagePath'] = packagePath;
// set an Azure DevOps variable to the resolved version
if (common.isLatest(packageVersion)) {
tl.setVariable(getLatestEnviromentVariable(packageName, common.isLatestPreRelease(packageVersion)), resolvedVersion);
}
}
}
return response;
}
/**
* Generates a unique environment variable for a nuget package name for it's latest version
*
* @param packageName - The name of the package to generate an environment variable for
* @returns The environment variable name
*/
function getLatestEnviromentVariable(packageName: string, isPreRelease: boolean): string {
let suffix = isPreRelease ? '_LATESTPRERELEASEVERSION' : '_LATESTVERSION';
return `MSDO_${packageName.replace(/\./g, '').replace('-', '')}${suffix}`.toUpperCase();
}
/**
* Checks if a NuGet package is installed.
*
* @param packageName - The name of the package to check
* @param packageVersion - The version of the package to check
* @param outputDirectory - The directory to check for the package
* @returns An object with information about the installed package.
*/
async function getInstallationStatus(
packageName: string,
packageVersion: string,
outputDirectory: string,
force: boolean = false): Promise<InstallNuGetPackageResponse> {
let response = {
success: false,
inCache: false,
packageName: packageName,
packageVersion: packageVersion
};
let checkInstall = true;
if (!force) {
// If latest, see if it's already been installed in a previous build step
const isLatest = common.isLatest(packageVersion);
// Only check if an exact version is installed
checkInstall = !isLatest;
if (isLatest) {
const isLatestPreRelease = common.isLatestPreRelease(packageVersion);
let cachedVersion = tl.getVariable(getLatestEnviromentVariable(packageName, isLatestPreRelease));
if (!common.isNullOrWhiteSpace(cachedVersion)) {
packageVersion = cachedVersion;
checkInstall = true;
}
}
}
if (checkInstall) {
const packagePath = getNuGetPackageFilePath(packageName, packageVersion, outputDirectory);
const packageFolder = common.removeExtension(packagePath);
const packageFolderExists = await common.directoryExists(packageFolder);
if (packageFolderExists) {
response['success'] = true;
response['inCache'] = true;
response['resolvedVersion'] = packageVersion;
response['packageFolder'] = packageFolder;
response['packagePath'] = packagePath;
}
}
return response;
}
/**
* Top level service call to resolve a package version from a NuGet server.
*
* @param serviceIndex - The response from calling the index.json entry point of a NuGet server
* @param requestOptions - The request options to use when calling the NuGet server, including authentication
* @param packageName - The name of the package to resolve
* @param packageVersion - The version of the package to resolve, accepts exact values or "Latest" and "LatestPreRelease" respectively.
* @returns If an exact version is requested, it will be returned. If the requested version is undefined, null, 'latest' or 'latest-prerelease', it calls the NuGet Server's SearchQueryService to find a real version.
*/
async function resolveVersion(
serviceIndex: Object,
requestOptions: Object,
packageName: string,
packageVersion: string): Promise<string> {
let resolvedVersion = packageVersion;
if (common.isLatest(packageVersion)) {
let serviceResponse: FindServiceResponse = findService(serviceIndex, 'RegistrationsBaseUrl', ['3.6.0', '3.0.0-beta']);
let serviceOptions = {
packageName: packageName,
packageVersion: packageVersion
};
resolvedVersion = await callService(serviceResponse, requestOptions, serviceOptions, _resolveVersion);
}
tl.debug(`resolvedVersion = ${resolvedVersion}`);
return resolvedVersion;
}
/**
* Business logic to call a single SearchQueryService endpoint to resolve a package version.
*
* @param service - The service to call
* @param serviceOptions - Input options boxed in an object to be wrapped around multiple calls for service resiliency
* @returns The resolved version of the package.
*/
async function _resolveVersion(
service: NuGetServiceResource,
requestOptions: Object,
serviceOptions: Object): Promise<any> {
// unbox input parmaeters
let packageName = serviceOptions['packageName'];
let packageVersion = serviceOptions['packageVersion'];
let resolvedVersion = null;
let searchQueryServiceUrlWithQuery = `${service['@id']}${packageName.toLowerCase()}/index.json`;
let result = await requestJson(searchQueryServiceUrlWithQuery, requestOptions);
const findPreRelease = common.isLatestPreRelease(packageVersion);
resolvedVersion = findLatestVersion(result, findPreRelease);
if (resolvedVersion == null) {
throw new Error(`Package not found: ${packageName}`);
}
return resolvedVersion;
}
/**
* Finds the latest version in a SearchQueryService response.
*
* @param result A JSON object returned from the SearchQueryService
* @param findPreRelease Whether or not to find a pre-release version
* @returns The latest version of the package
*/
function findLatestVersion(
result: object,
findPreRelease: boolean): string {
let latestVersion = null;
let latestVersionParts = null;
let latestIsPreRelease = false;
let latestPreReleaseFlag = null;
if (result == null || result['items'] == null) {
return latestVersion;
}
let currentCatalogEntry = null;
let currentVersion = null;
let currentVersionParts = null;
let currentFullVersionParts = null;
let currentVersionNumbersString = null;
let currentIsLatest = false;
let currentIsPreRelease = false;
let currentPreReleaseFlag = null;
for (let packageGroup of result['items']) {
for (let packageInfo of packageGroup['items']) {
currentCatalogEntry = packageInfo['catalogEntry'];
if (currentCatalogEntry['listed'] != true) {
// skip delisted packages
continue;
}
currentVersion = currentCatalogEntry['version'];
currentIsPreRelease = common.isPreRelease(currentVersion);
if (!findPreRelease && currentIsPreRelease) {
// skip prerelease packages if we're looking for a stable version
continue;
}
currentFullVersionParts = currentVersion.split("-");
if (currentIsPreRelease) {
currentPreReleaseFlag = currentFullVersionParts[1];
}
currentVersionNumbersString = currentFullVersionParts[0];
currentVersionParts = currentVersionNumbersString.split(".");
currentIsLatest = latestVersion == null;
if (!currentIsLatest) {
// Evaluate the current version against the latest version
// Handle comparisons of separate level versions
// Some packages exclude Patch or include Revisions up to two levels (Rev1 and Rev2)
let maxVersionParts = currentVersionParts.length;
if (currentVersionParts.length < maxVersionParts) {
maxVersionParts = latestVersionParts.length;
}
for (let versionPartIndex = 0; versionPartIndex < currentVersionParts.length; versionPartIndex++) {
let versionPart = 0;
let latestVersionPart = 0;
let isLastVersionPart = versionPartIndex == (maxVersionParts - 1);
if (versionPartIndex < currentVersionParts.length) {
versionPart = parseInt(currentVersionParts[versionPartIndex]);
}
if (versionPartIndex < latestVersionParts.length) {
latestVersionPart = parseInt(latestVersionParts[versionPartIndex]);
}
if (versionPart > latestVersionPart) {
currentIsLatest = true;
} else if (versionPart == latestVersionPart) {
currentIsLatest = isLastVersionPart
&&
(
(currentIsPreRelease && latestIsPreRelease && currentPreReleaseFlag > latestPreReleaseFlag)
||
(!currentIsPreRelease && latestIsPreRelease)
);
} else {
// Current version is less than latest found
break;
}
if (currentIsLatest) {
break;
}
}
}
if (currentIsLatest) {
latestVersion = currentVersion;
latestVersionParts = currentVersionParts;
latestIsPreRelease = currentIsPreRelease;
latestPreReleaseFlag = currentPreReleaseFlag;
}
}
}
tl.debug(`latestVersion = ${latestVersion}`);
return latestVersion;
}
// function rampedDeployment(
// datetime: Date,
// rampMinutes: number): boolean {
// let ramped = false;
// let curDate = new Date();
// let diff = curDate.getTime() - datetime.getTime();
// datetime.setMinutes
// return Math.random() > diff;
// }
/**
* Top level service call to download a package from a NuGet server.
*
* @param serviceIndex - The response from calling the index.json entry point of a NuGet server
* @param requestOptions - The request options to use when calling the NuGet server, including authentication
* @param packageName - The name of the package to download
* @param resolvedVersion - The version of the package to download
* @param outputDirectory - The directory to download the package to
* @returns The path to the downloaded package
*/
async function downloadPackage(
serviceIndex: Object,
requestOptions: Object,
packageName: string,
resolvedVersion: string,
outputDirectory: string): Promise<string> {
let serviceResponse: FindServiceResponse = findService(serviceIndex, 'PackageBaseAddress', ['3.0.0']);
let serviceOptions = {
packageName: packageName,
resolvedVersion: resolvedVersion,
outputDirectory: outputDirectory
};
return await callService(serviceResponse, requestOptions, serviceOptions, _downloadPackage);
}
/**
* Business logic to call download a package from a NuGet server.
*
* @param service - The service to call
* @param serviceOptions - Input options boxed in an object to be wrapped around multiple calls for service resiliency
* @returns The path to the downloaded package
*/
async function _downloadPackage(
service: NuGetServiceResource,
requestOptions: Object,
serviceOptions: Object): Promise<any> {
// unbox input parameters
const packageName = serviceOptions['packageName'];
const resolvedVersion = serviceOptions['resolvedVersion'];
const outputDirectory = serviceOptions['outputDirectory'];
const packageNameLower = packageName.toLowerCase();
const resolvedVersionLower = resolvedVersion.toLowerCase();
const packageUrl = `${service['@id']}${packageNameLower}/${resolvedVersionLower}/${packageNameLower}.${resolvedVersionLower}.nupkg`;
const destinationPath = getNuGetPackageFilePath(packageName, resolvedVersion, outputDirectory);
await downloadFile(packageUrl, requestOptions, destinationPath);
// if the package is not found, throw an error
if (!fs.existsSync(destinationPath)) {
throw new Error(`The package could not be found after download: ${destinationPath}`);
}
return destinationPath;
}
/**
* Given a package name and version, returns the path to the downloaded package.
*
* @param packageName - The name of the package to download
* @param packageVersion - The version of the package to download
* @param outputDirectory - The directory to download the package to
* @returns A path to the downloaded package
*/
function getNuGetPackageFilePath(
packageName: string,
packageVersion: string,
outputDirectory: string): string {
return path.join(outputDirectory, `${packageName}.${packageVersion}.nupkg`);
}
/**
* Extracts the input file path to a directory matching the file name without the extension.
*
* @param filePath - The path to the package to extract
*/
async function extractPackage(filePath: string): Promise<void> {
let packageDirectory = common.removeExtension(filePath);
let zip = new AdmZip(filePath);
zip.extractAllTo(packageDirectory, true);
await enableOnLinux(packageDirectory);
}
/**
* Given the input service index response of a NuGet server, find all services of the given service name.
*
* @param serviceIndex - The response from calling the index.json entry point of a NuGet server
* @param serviceName - The name of the service to find
* @param knownServiceVersions - Versions of the service we know about
*/
function findService(
serviceIndex: Object,
serviceName: string,
knownServiceVersions: string[]): FindServiceResponse {
// initialize the response
const response: FindServiceResponse = {
known: [],
unknown: []
};
for (const service of serviceIndex["resources"]) {
const serviceParts = service['@type'].split('/');
if (serviceParts === undefined || serviceParts.length !== 2) {
// skip this service
continue;
}
const _serviceName = serviceParts[0];
const _serviceVersion = serviceParts[1];
if (_serviceName === serviceName) {
// create the service response
// splitting out name and version for later processing
// it will either be added to known or unknown
const serviceResponse = {
'@id': service['@id'],
'@type': service['@type'],
'name': _serviceName,
'version': _serviceVersion
};
// see if we know about this version
if (knownServiceVersions.indexOf(_serviceVersion) > -1) {
response.known.push(serviceResponse);
} else {
response.unknown.push(serviceResponse);
}
}
}
if (response.known.length === 0 && response.unknown.length === 0) {
throw new Error(`Could not find service: ${serviceName}`);
}
return response;
}
/**
* Interface for the business logic calls to services to be called by the
* callService wrapper function to ensure service resiliency.
*/
interface ServiceVersionCalls {
[version: string]: (service: NuGetServiceResource, requestOptions: Object, serviceOptions: Object) => Promise<any>;
}
/**
* Calls all known service versions until one succeeds.
* If none of those succeeds, it attempts to call all unknown service versions.
* This is to provide service resiliency, allowing the server to make version updates
* without breaking the client code.
* It also provides prioritization of known versions over unknown versions.
*
* @param serviceResponse - The response from calling the findService function
* @param requestOptions - The request options to use when calling the NuGet server, including authentication
* @param serviceOptions - Input options boxed in an object to be wrapped around multiple calls for service resiliency
* @param serviceCall - The business logic call to the service
* @param serviceVersionCalls - The business logic calls to the service for each known version
* @returns
*/
async function callService(
serviceResponse: FindServiceResponse,
requestOptions: Object,
serviceOptions: Object,
serviceCall: (service: NuGetServiceResource, requestOptions: Object, serviceOptions: Object) => Promise<any>,
serviceVersionCalls: ServiceVersionCalls = null): Promise<any> {
let response: any;
let services = serviceResponse.known;
let isKnown = true;
if (services === undefined || services.length === 0) {
services = serviceResponse.unknown;
isKnown = false;
}
let firstError: Error;
let i = 0;
// try each known service until we find one that works
// then try each unknown service until we find one that works
do {
try {
const service = services[i];
let _serviceCall: (service: NuGetServiceResource, requestOptions: Object, serviceOptions: Object) => Promise<any> = serviceCall;
if (serviceVersionCalls != null && serviceVersionCalls[service['version']] !== undefined) {
_serviceCall = serviceVersionCalls[service['version']];
}
response = await _serviceCall(service, requestOptions, serviceOptions);
break;
} catch (error) {
tl.debug(`Failed to call service: ${error.message}`);
if (firstError === undefined) {
firstError = error;
}
i += 1;
if (i == services.length) {
if (isKnown) {
isKnown = false;
// try unknown services
tl.debug('Attempting to call unknown service type versions...');
services = serviceResponse.unknown;
if (services === undefined || services.length === 0) {
throw firstError;
}
i = 0;
} else {
// rethrow the first error we saw
// if known services were attempted, it will throw the error from the known service
throw firstError;
}
}
}
} while (true)
return response;
}
/**
* Resolves https request options.
*
* @param accessToken - (Optional) The access token to use when calling the NuGet server
* @returns Https request options.
*/
function resolveRequestOptions(accessToken: string): Object {
// Get proxy settings
const proxyConfig = tl.getHttpProxyConfiguration();
const proxyAgent = proxyConfig ? new HttpsProxyAgent(proxyConfig.proxyFormattedUrl) : null;
let options = {
method: 'GET',
timeout: 2500,
headers: {
'Content-Type': 'application/json'
},
agent: proxyAgent
};
if (!common.isNullOrWhiteSpace(accessToken)) {
options['auth'] = `:${accessToken}`;
}
return options;
}
/**
* Calls an https endpoint and returns the response as a JSON object.
*
* @param url - The url to call
* @param options - The request options to use when calling the NuGet server, including authentication
* @returns The response as a JSON object.
*/
async function requestJson(url: string, options: Object): Promise<Object> {
return new Promise((resolve, reject) => {
tl.debug(`${options['method'].toUpperCase()} ${url}`);
const req = https.request(url, options, async (res) => {
// attempt to decompress the response if it's gzipped
try {
const decompressResponse = await import('decompress-response');
res = decompressResponse.default(res);
} catch (error) {
tl.debug(`Failed to add response decompression: ${error.message}`);
}
if (res.statusCode !== 200) {
reject(new Error(`Failed to call: ${url}. Status code: ${res.statusCode}`));
return;
}
let data = '';
res.on('data', (chunk) => {
data += chunk.toString();
});
res.on('end', () => {
let jsonData;
try {
jsonData = JSON.parse(data);
} catch (error) {
reject(new Error(`Failed to parse JSON: ${data}`));
return;
}
resolve(jsonData);
});
});
req.on('error', (error) => {
reject(new Error(`Error calling url: ${error}`));
});
req.end();
});
}
/**
* Downloads a file from a url.
* Will follow 303 redirects.
*
* @param url - The url to download the file from
* @param options - The request options to use when calling the NuGet server, including authentication
* @param destinationPath - The path to download the file to
*/
async function downloadFile(
url: string,
options: Object,
destinationPath: string,
retries: number = _defaultFileDownloadRetries,
retryDelay: number = _defaultFileDownloadRetryDelayMs): Promise<void> {
return new Promise(async (resolve, reject) => {
let errors: Error[] = [];
do {
try {
await _downloadFile(url, options, destinationPath);
resolve();
return;
} catch (error) {
errors.push(error);
if (retries > 0) {
tl.debug(`Error downloading url: ${error.message}`);
tl.debug(`Retrying download of url: ${url}`);
await common.sleep(retryDelay);
}
}
} while (retries-- > 0);
reject(new Error(`Error downloading url: ${errors[0] || url}`));
});
}
/**
* Downloads a file from a url.
* Will follow 303 redirects.
*
* @param url - The url to download the file from
* @param options - The request options to use when calling the NuGet server, including authentication
* @param destinationPath - The path to download the file to
*/
async function _downloadFile(
url: string,
options: Object,
destinationPath: string): Promise<void> {
return new Promise(async (resolve, reject) => {
const req = https.request(url, options, async (res) => {
if (res.statusCode === 303) {
let redirectUrl = res.headers['location'];
options['auth'] = null;
await downloadFile(redirectUrl, options, destinationPath);
resolve();
return;
}
if (res.statusCode !== 200) {
reject(`Failed to download file: ${url}. Status code: ${res.statusCode}`);
return;
}
const file = fs.createWriteStream(destinationPath);
res.pipe(file);
file.on('finish', () => {
file.close();
resolve();
});
});
req.on('error', (error) => {
reject(error);
});
req.end();
});
}
/**
* Runs chmod 0o755 on all files within the folder if the platform is "linux"
*
* @param folderPath - The path to the folder to enable executables on
*/
async function enableOnLinux(folderPath: string): Promise<void> {
return new Promise(async (resolve, reject) => {
if (process.platform != 'linux') {
resolve();
return;
}
const entries = fs.readdirSync(folderPath);
const tasks = entries.map(async (entry) => {
try {
const entryPath = path.join(folderPath, entry);
const stats = fs.statSync(entryPath);
if (stats.isFile()) {
try {
fs.chmodSync(entryPath, 0o755);
tl.debug(`0o755 permission set for: ${entryPath}`);
} catch (error) {
tl.debug(`Error setting executable permission: ${error.message}`);
}
} else if (stats.isDirectory()) {
await enableOnLinux(entryPath);
}
} catch (error) {
reject(new Error(`Error getting file stats: ${error.message}`));
}
});
await Promise.all(tasks);
resolve();
});
}