@@ -12,13 +12,14 @@ import * as semver from "semver"
12
12
import * as vscode from "vscode"
13
13
import { makeCoderSdk , startWorkspace , waitForBuild } from "./api"
14
14
import { extractAgents } from "./api-helper"
15
+ import * as cli from "./cliManager"
15
16
import { Commands } from "./commands"
17
+ import { featureSetForVersion , FeatureSet } from "./featureSet"
16
18
import { getHeaderCommand } from "./headers"
17
19
import { SSHConfig , SSHValues , mergeSSHConfigValues } from "./sshConfig"
18
20
import { computeSSHProperties , sshSupportsSetEnv } from "./sshSupport"
19
21
import { Storage } from "./storage"
20
22
import { AuthorityPrefix , expandPath , parseRemoteAuthority } from "./util"
21
- import { supportsCoderAgentLogDirFlag } from "./version"
22
23
import { WorkspaceAction } from "./workspaceAction"
23
24
24
25
export interface RemoteDetails extends vscode . Disposable {
@@ -33,7 +34,6 @@ export class Remote {
33
34
private readonly storage : Storage ,
34
35
private readonly commands : Commands ,
35
36
private readonly mode : vscode . ExtensionMode ,
36
- private coderVersion : semver . SemVer | null = null ,
37
37
) { }
38
38
39
39
private async confirmStart ( workspaceName : string ) : Promise < boolean > {
@@ -194,16 +194,34 @@ export class Remote {
194
194
// Store for use in commands.
195
195
this . commands . workspaceRestClient = workspaceRestClient
196
196
197
+ let binaryPath : string | undefined
198
+ if ( this . mode === vscode . ExtensionMode . Production ) {
199
+ binaryPath = await this . storage . fetchBinary ( workspaceRestClient , parts . label )
200
+ } else {
201
+ try {
202
+ // In development, try to use `/tmp/coder` as the binary path.
203
+ // This is useful for debugging with a custom bin!
204
+ binaryPath = path . join ( os . tmpdir ( ) , "coder" )
205
+ await fs . stat ( binaryPath )
206
+ } catch ( ex ) {
207
+ binaryPath = await this . storage . fetchBinary ( workspaceRestClient , parts . label )
208
+ }
209
+ }
210
+
197
211
// First thing is to check the version.
198
212
const buildInfo = await workspaceRestClient . getBuildInfo ( )
199
- this . coderVersion = semver . parse ( buildInfo . version )
213
+
214
+ let version : semver . SemVer | null = null
215
+ try {
216
+ version = semver . parse ( await cli . version ( binaryPath ) )
217
+ } catch ( e ) {
218
+ version = semver . parse ( buildInfo . version )
219
+ }
220
+
221
+ const featureSet = featureSetForVersion ( version )
222
+
200
223
// Server versions before v0.14.1 don't support the vscodessh command!
201
- if (
202
- this . coderVersion ?. major === 0 &&
203
- this . coderVersion ?. minor <= 14 &&
204
- this . coderVersion ?. patch < 1 &&
205
- this . coderVersion ?. prerelease . length === 0
206
- ) {
224
+ if ( ! featureSet . vscodessh ) {
207
225
await this . vscodeProposed . window . showErrorMessage (
208
226
"Incompatible Server" ,
209
227
{
@@ -501,7 +519,7 @@ export class Remote {
501
519
// "Host not found".
502
520
try {
503
521
this . storage . writeToCoderOutputChannel ( "Updating SSH config..." )
504
- await this . updateSSHConfig ( workspaceRestClient , parts . label , parts . host )
522
+ await this . updateSSHConfig ( workspaceRestClient , parts . label , parts . host , binaryPath , featureSet )
505
523
} catch ( error ) {
506
524
this . storage . writeToCoderOutputChannel ( `Failed to configure SSH: ${ error } ` )
507
525
throw error
@@ -544,8 +562,8 @@ export class Remote {
544
562
/**
545
563
* Format's the --log-dir argument for the ProxyCommand
546
564
*/
547
- private async formatLogArg ( ) : Promise < string > {
548
- if ( ! supportsCoderAgentLogDirFlag ( this . coderVersion ) ) {
565
+ private async formatLogArg ( featureSet : FeatureSet ) : Promise < string > {
566
+ if ( ! featureSet . proxyLogDirectory ) {
549
567
return ""
550
568
}
551
569
@@ -563,7 +581,13 @@ export class Remote {
563
581
564
582
// updateSSHConfig updates the SSH configuration with a wildcard that handles
565
583
// all Coder entries.
566
- private async updateSSHConfig ( restClient : Api , label : string , hostName : string ) {
584
+ private async updateSSHConfig (
585
+ restClient : Api ,
586
+ label : string ,
587
+ hostName : string ,
588
+ binaryPath : string ,
589
+ featureSet : FeatureSet ,
590
+ ) {
567
591
let deploymentSSHConfig = { }
568
592
try {
569
593
const deploymentConfig = await restClient . getDeploymentSSHConfig ( )
@@ -624,20 +648,6 @@ export class Remote {
624
648
const sshConfig = new SSHConfig ( sshConfigFile )
625
649
await sshConfig . load ( )
626
650
627
- let binaryPath : string | undefined
628
- if ( this . mode === vscode . ExtensionMode . Production ) {
629
- binaryPath = await this . storage . fetchBinary ( restClient , label )
630
- } else {
631
- try {
632
- // In development, try to use `/tmp/coder` as the binary path.
633
- // This is useful for debugging with a custom bin!
634
- binaryPath = path . join ( os . tmpdir ( ) , "coder" )
635
- await fs . stat ( binaryPath )
636
- } catch ( ex ) {
637
- binaryPath = await this . storage . fetchBinary ( restClient , label )
638
- }
639
- }
640
-
641
651
const escape = ( str : string ) : string => `"${ str . replace ( / " / g, '\\"' ) } "`
642
652
// Escape a command line to be executed by the Coder binary, so ssh doesn't substitute variables.
643
653
const escapeSubcommand : ( str : string ) => string =
@@ -659,7 +669,7 @@ export class Remote {
659
669
Host : label ? `${ AuthorityPrefix } .${ label } --*` : `${ AuthorityPrefix } --*` ,
660
670
ProxyCommand : `${ escape ( binaryPath ) } ${ headerArg } vscodessh --network-info-dir ${ escape (
661
671
this . storage . getNetworkInfoPath ( ) ,
662
- ) } ${ await this . formatLogArg ( ) } --session-token-file ${ escape ( this . storage . getSessionTokenPath ( label ) ) } --url-file ${ escape (
672
+ ) } ${ await this . formatLogArg ( featureSet ) } --session-token-file ${ escape ( this . storage . getSessionTokenPath ( label ) ) } --url-file ${ escape (
663
673
this . storage . getUrlPath ( label ) ,
664
674
) } %h`,
665
675
ConnectTimeout : "0" ,
0 commit comments