Skip to content

Commit c86868f

Browse files
authoredNov 5, 2024··
Consider Runtime Patch Versions in PATH API (#2008)
* Consider Runtime Patch Versions in PATH API The C# extension will now fail if the runtime is not 8.0.10 or higher on mac so we need to support patch version lookup. Sadly we cannot migrate to semver because semver rejects strings such as 8.0 which are allowable in many parts of our code and APIs. This also adds code for SDK lookup but no test since that's harder to test, like remarked in the code. * Fix test * Fix the version logic to not only parse major.minor version We now parse all of the version, dont need to do this logic.
1 parent f02cc27 commit c86868f

File tree

5 files changed

+98
-22
lines changed

5 files changed

+98
-22
lines changed
 

‎vscode-dotnet-runtime-extension/src/test/functional/DotnetCoreAcquisitionExtension.test.ts

+12-1
Original file line numberDiff line numberDiff line change
@@ -247,7 +247,7 @@ suite('DotnetCoreAcquisitionExtension End to End', function()
247247
}
248248
else
249249
{
250-
assert.equal(result, undefined, 'find path command returned no undefined if no path matches condition');
250+
assert.equal(result?.dotnetPath, undefined, 'find path command returned no undefined if no path matches condition');
251251
}
252252
}
253253

@@ -379,6 +379,17 @@ suite('DotnetCoreAcquisitionExtension End to End', function()
379379
}
380380
}).timeout(standardTimeoutTime);
381381

382+
test('Find dotnet PATH Command Unmet Runtime Patch Condition', async () => {
383+
// Install 8.0.{LATEST, which will be < 99}, look for 8.0.99 with accepting dotnet gr than or eq to 8.0.99
384+
// No tests for SDK since that's harder to replicate with a global install and different machine states
385+
if(os.platform() !== 'darwin')
386+
{
387+
await findPathWithRequirementAndInstall('8.0', 'runtime', os.arch(), 'greater_than_or_equal', false,
388+
{version : '8.0.99', mode : 'runtime', architecture : os.arch(), requestingExtensionId : requestingExtensionId}
389+
);
390+
}
391+
}).timeout(standardTimeoutTime);
392+
382393
test('Install SDK Globally E2E (Requires Admin)', async () => {
383394
// We only test if the process is running under ADMIN because non-admin requires user-intervention.
384395
if(new FileUtilities().isElevated())

‎vscode-dotnet-runtime-library/src/Acquisition/DotnetConditionValidator.ts

+34-16
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,6 @@
22
* Licensed to the .NET Foundation under one or more agreements.
33
* The .NET Foundation licenses this file to you under the MIT license.
44
*--------------------------------------------------------------------------------------------*/
5-
import { DotnetVersionSpecRequirement } from '../DotnetVersionSpecRequirement';
65
import { IDotnetFindPathContext } from '../IDotnetFindPathContext';
76
import { CommandExecutor } from '../Utils/CommandExecutor';
87
import { ICommandExecutor } from '../Utils/ICommandExecutor';
@@ -26,14 +25,12 @@ export class DotnetConditionValidator implements IDotnetConditionValidator
2625
public async dotnetMeetsRequirement(dotnetExecutablePath: string, requirement : IDotnetFindPathContext) : Promise<boolean>
2726
{
2827
const availableRuntimes = await this.getRuntimes(dotnetExecutablePath);
29-
const requestedMajorMinor = versionUtils.getMajorMinor(requirement.acquireContext.version, this.workerContext.eventStream, this.workerContext);
3028
const hostArch = await this.getHostArchitecture(dotnetExecutablePath, requirement);
3129

3230
if(availableRuntimes.some((runtime) =>
3331
{
34-
const availableVersion = versionUtils.getMajorMinor(runtime.version, this.workerContext.eventStream, this.workerContext);
3532
return runtime.mode === requirement.acquireContext.mode && this.stringArchitectureMeetsRequirement(hostArch, requirement.acquireContext.architecture) &&
36-
this.stringVersionMeetsRequirement(availableVersion, requestedMajorMinor, requirement.versionSpecRequirement);
33+
this.stringVersionMeetsRequirement(runtime.version, requirement.acquireContext.version, requirement);
3734
}))
3835
{
3936
return true;
@@ -44,16 +41,15 @@ export class DotnetConditionValidator implements IDotnetConditionValidator
4441
if(availableSDKs.some((sdk) =>
4542
{
4643
// The SDK includes the Runtime, ASP.NET Core Runtime, and Windows Desktop Runtime. So, we don't need to check the mode.
47-
const availableVersion = versionUtils.getMajorMinor(sdk.version, this.workerContext.eventStream, this.workerContext);
48-
return this.stringArchitectureMeetsRequirement(hostArch, requirement.acquireContext.architecture) && this.stringVersionMeetsRequirement(availableVersion, requestedMajorMinor, requirement.versionSpecRequirement);
44+
return this.stringArchitectureMeetsRequirement(hostArch, requirement.acquireContext.architecture) && this.stringVersionMeetsRequirement(sdk.version, requirement.acquireContext.version, requirement);
4945
}))
5046
{
5147
return true;
5248
}
5349
else
5450
{
5551
this.workerContext.eventStream.post(new DotnetFindPathDidNotMeetCondition(`${dotnetExecutablePath} did NOT satisfy the conditions: hostArch: ${hostArch}, requiredArch: ${requirement.acquireContext.architecture},
56-
required version: ${requestedMajorMinor}`));
52+
required version: ${requirement.acquireContext.version}, required mode: ${requirement.acquireContext.mode}`));
5753
}
5854
}
5955

@@ -136,29 +132,51 @@ Please set the PATH to a dotnet host that matches the architecture ${requirement
136132
return os.platform() === 'win32' ? (await this.executor!.tryFindWorkingCommand([CommandExecutor.makeCommand('chcp', ['65001'])])) !== null : false;
137133
}
138134

139-
private stringVersionMeetsRequirement(availableVersion : string, requestedVersion : string, requirement : DotnetVersionSpecRequirement) : boolean
135+
private stringVersionMeetsRequirement(availableVersion : string, requestedVersion : string, requirement : IDotnetFindPathContext) : boolean
140136
{
141137
const availableMajor = Number(versionUtils.getMajor(availableVersion, this.workerContext.eventStream, this.workerContext));
142138
const requestedMajor = Number(versionUtils.getMajor(requestedVersion, this.workerContext.eventStream, this.workerContext));
139+
const requestedPatchStr : string | null = requirement.acquireContext.mode !== 'sdk' ? versionUtils.getRuntimePatchVersionString(requestedVersion, this.workerContext.eventStream, this.workerContext)
140+
: versionUtils.getSDKCompleteBandAndPatchVersionString(requestedVersion, this.workerContext.eventStream, this.workerContext);
141+
const requestedPatch = requestedPatchStr ? Number(requestedPatchStr) : null;
143142

144143
if(availableMajor === requestedMajor)
145144
{
146145
const availableMinor = Number(versionUtils.getMinor(availableVersion, this.workerContext.eventStream, this.workerContext));
147146
const requestedMinor = Number(versionUtils.getMinor(requestedVersion, this.workerContext.eventStream, this.workerContext));
148147

149-
switch(requirement)
148+
if(availableMinor === requestedMinor && requestedPatch)
150149
{
151-
case 'equal':
152-
return availableMinor === requestedMinor;
153-
case 'greater_than_or_equal':
154-
return availableMinor >= requestedMinor;
155-
case 'less_than_or_equal':
156-
return availableMinor <= requestedMinor;
150+
const availablePatchStr : string | null = requirement.acquireContext.mode !== 'sdk' ? versionUtils.getRuntimePatchVersionString(availableVersion, this.workerContext.eventStream, this.workerContext)
151+
: versionUtils.getSDKCompleteBandAndPatchVersionString(availableVersion, this.workerContext.eventStream, this.workerContext);
152+
const availablePatch = availablePatchStr ? Number(availablePatchStr) : null;
153+
switch(requirement.versionSpecRequirement)
154+
{
155+
case 'equal':
156+
return availablePatch === requestedPatch;
157+
case 'greater_than_or_equal':
158+
// the 'availablePatch' must exist, since the version is from --list-runtimes or --list-sdks.
159+
return availablePatch! >= requestedPatch;
160+
case 'less_than_or_equal':
161+
return availablePatch! <= requestedPatch;
162+
}
163+
}
164+
else
165+
{
166+
switch(requirement.versionSpecRequirement)
167+
{
168+
case 'equal':
169+
return availableMinor === requestedMinor;
170+
case 'greater_than_or_equal':
171+
return availableMinor >= requestedMinor;
172+
case 'less_than_or_equal':
173+
return availableMinor <= requestedMinor;
174+
}
157175
}
158176
}
159177
else
160178
{
161-
switch(requirement)
179+
switch(requirement.versionSpecRequirement)
162180
{
163181
case 'equal':
164182
return false;

‎vscode-dotnet-runtime-library/src/Acquisition/VersionUtilities.ts

+35-5
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@
55

66
import { IAcquisitionWorkerContext } from './IAcquisitionWorkerContext';
77
import { IEventStream } from '../EventStream/EventStream';
8-
import { DotnetFeatureBandDoesNotExistError, DotnetVersionParseEvent, DotnetVersionResolutionError, EventCancellationError } from '../EventStream/EventStreamEvents';
8+
import { DotnetFeatureBandDoesNotExistError, DotnetInvalidRuntimePatchVersion, DotnetVersionParseEvent, DotnetVersionResolutionError, EventCancellationError } from '../EventStream/EventStreamEvents';
99
import { getInstallFromContext } from '../Utils/InstallIdUtilities';
1010

1111
const invalidFeatureBandErrorString = `A feature band couldn't be determined for the requested version: `;
@@ -78,15 +78,15 @@ export function getFeatureBandFromVersion(fullySpecifiedVersion : string, eventS
7878
*/
7979
export function getFeatureBandPatchVersion(fullySpecifiedVersion : string, eventStream : IEventStream, context : IAcquisitionWorkerContext) : string
8080
{
81-
return Number(getPatchVersionString(fullySpecifiedVersion, eventStream, context)).toString();
81+
return Number(getSDKPatchVersionString(fullySpecifiedVersion, eventStream, context)).toString();
8282
}
8383

8484
/**
8585
*
8686
* @remarks the logic for getFeatureBandPatchVersion, except that it returns '01' or '00' instead of the patch number.
8787
* Not meant for public use.
8888
*/
89-
export function getPatchVersionString(fullySpecifiedVersion : string, eventStream : IEventStream, context : IAcquisitionWorkerContext) : string
89+
export function getSDKPatchVersionString(fullySpecifiedVersion : string, eventStream : IEventStream, context : IAcquisitionWorkerContext) : string
9090
{
9191
const patch : string | undefined = fullySpecifiedVersion.split('.')?.at(2)?.substring(1)?.split('-')?.at(0);
9292
if(patch === undefined || !isNumber(patch))
@@ -100,6 +100,36 @@ export function getPatchVersionString(fullySpecifiedVersion : string, eventStrea
100100
return patch
101101
}
102102

103+
/**
104+
*
105+
* @param fullySpecifiedVersion the version of the sdk, either fully specified or not, but containing a band definition.
106+
* @returns a single string representing the band and patch version, e.g. 312 in 7.0.312.
107+
*/
108+
export function getSDKCompleteBandAndPatchVersionString(fullySpecifiedVersion : string, eventStream : IEventStream, context : IAcquisitionWorkerContext) : string
109+
{
110+
return `${getFeatureBandFromVersion(fullySpecifiedVersion, eventStream, context)}${getSDKPatchVersionString(fullySpecifiedVersion, eventStream, context)}`;
111+
}
112+
113+
/**
114+
* The runtime version doesn't have a feature band, unlike the SDK. We need to get the patch version from the runtime version.
115+
* It can contain any amount of text after the patch, such as 9.0.0-rc.2.24473.5. We don't process any of that extra text and should ignore it.
116+
* Needs to handle 8, 8.0, etc. This is why we don't use semver. We don't error if there isn't a patch but we should if the patch is invalid.
117+
* Returns null if no patch is in the string (e.g. 8.0).
118+
*/
119+
export function getRuntimePatchVersionString(fullySpecifiedVersion : string, eventStream : IEventStream, context : IAcquisitionWorkerContext) : string | null
120+
{
121+
const patch : string | undefined = fullySpecifiedVersion.split('.')?.at(2)?.split('-')?.at(0);
122+
if(patch && !isNumber(patch))
123+
{
124+
const event = new DotnetInvalidRuntimePatchVersion(new EventCancellationError('DotnetInvalidRuntimePatchVersion',
125+
`The runtime patch version ${patch} from ${fullySpecifiedVersion} is NaN.`),
126+
getInstallFromContext(context));
127+
eventStream.post(event);
128+
throw event.error;
129+
}
130+
return patch ? patch : null;
131+
}
132+
103133
/**
104134
*
105135
* @param fullySpecifiedVersion the requested version to analyze.
@@ -113,8 +143,8 @@ export function isValidLongFormVersionFormat(fullySpecifiedVersion : string, eve
113143
{
114144
if(isNonSpecificFeatureBandedVersion(fullySpecifiedVersion) ||
115145
(
116-
getPatchVersionString(fullySpecifiedVersion, eventStream, context).length <= 2 &&
117-
getPatchVersionString(fullySpecifiedVersion, eventStream, context).length > 1
146+
getSDKPatchVersionString(fullySpecifiedVersion, eventStream, context).length <= 2 &&
147+
getSDKPatchVersionString(fullySpecifiedVersion, eventStream, context).length > 1
118148
)
119149
)
120150
{

‎vscode-dotnet-runtime-library/src/EventStream/EventStreamEvents.ts

+4
Original file line numberDiff line numberDiff line change
@@ -487,6 +487,10 @@ export class DotnetFeatureBandDoesNotExistError extends DotnetAcquisitionError {
487487
public readonly eventName = 'DotnetFeatureBandDoesNotExistError';
488488
}
489489

490+
export class DotnetInvalidRuntimePatchVersion extends DotnetAcquisitionError {
491+
public readonly eventName = 'DotnetInvalidRuntimePatchVersion';
492+
}
493+
490494
export class DotnetWSLSecurityError extends DotnetInstallExpectedAbort {
491495
public readonly eventName = 'DotnetWSLSecurityError';
492496
}

‎vscode-dotnet-runtime-library/src/test/unit/VersionUtilities.test.ts

+13
Original file line numberDiff line numberDiff line change
@@ -59,6 +59,19 @@ suite('Version Utilities Unit Tests', () => {
5959
assert.equal(resolver.getFeatureBandPatchVersion(twoDigitPatchVersion, mockEventStream, mockCtx), '21');
6060
});
6161

62+
test('Get Band+Patch from SDK Version', async () => {
63+
assert.equal(resolver.getSDKCompleteBandAndPatchVersionString(fullySpecifiedVersion, mockEventStream, mockCtx), '201');
64+
assert.equal(resolver.getSDKCompleteBandAndPatchVersionString(uniqueMajorMinorVersion, mockEventStream, mockCtx), '300');
65+
assert.equal(resolver.getSDKCompleteBandAndPatchVersionString(twoDigitMajorVersion, mockEventStream, mockCtx), '102');
66+
assert.equal(resolver.getSDKCompleteBandAndPatchVersionString(twoDigitPatchVersion, mockEventStream, mockCtx), '221');
67+
});
68+
69+
test('Get Patch from Runtime Version', async () => {
70+
assert.equal(resolver.getRuntimePatchVersionString(majorMinorOnly, mockEventStream, mockCtx), null);
71+
assert.equal(resolver.getRuntimePatchVersionString('8.0.10', mockEventStream, mockCtx), '10');
72+
assert.equal(resolver.getRuntimePatchVersionString('8.0.9-rc.2.24502.A', mockEventStream, mockCtx), '9');
73+
});
74+
6275
test('Get Patch from SDK Preview Version', async () => {
6376
assert.equal(resolver.getFeatureBandPatchVersion('8.0.400-preview.0.24324.5', mockEventStream, mockCtx), '0');
6477
});

0 commit comments

Comments
 (0)
Please sign in to comment.