Skip to content

Commit

Permalink
fix(jetbrains): avoid connection loops
Browse files Browse the repository at this point in the history
  • Loading branch information
andreafalzetti committed Aug 30, 2022
1 parent db6f5ed commit 0b3d189
Show file tree
Hide file tree
Showing 2 changed files with 33 additions and 1 deletion.
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,8 @@ import com.intellij.openapi.diagnostic.thisLogger
import com.intellij.openapi.project.Project
import com.intellij.openapi.util.io.BufferExposingByteArrayOutputStream
import com.intellij.openapi.util.io.FileUtilRt
import com.intellij.openapi.wm.IdeFocusManager
import com.intellij.ui.AppIcon
import com.intellij.util.application
import com.intellij.util.withFragment
import com.intellij.util.withPath
Expand Down Expand Up @@ -101,6 +103,11 @@ class GitpodCLIService : RestService() {
ApplicationManager.getApplication().executeOnPooledThread {
getClientSessionAndProjectAsync().let { (session, project) ->
ClientId.withClientId(session.clientId) {
val toFocus = IdeFocusManager.getGlobalInstance()?.lastFocusedFrame
thisLogger().warn("toFocus = $toFocus")
if (toFocus != null) {
AppIcon.getInstance().requestFocus(toFocus)
}
action(project)
sendOk(request, context)
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ import com.intellij.ide.BrowserUtil
import com.intellij.openapi.components.service
import com.intellij.openapi.diagnostic.thisLogger
import com.intellij.openapi.progress.ProgressManager
import com.intellij.openapi.ui.Messages
import com.intellij.remote.RemoteCredentialsHolder
import com.intellij.ssh.AskAboutHostKey
import com.intellij.ssh.OpenSshLikeHostKeyVerifier
Expand All @@ -31,6 +32,7 @@ import com.jetbrains.gateway.ssh.ClientOverSshTunnelConnector
import com.jetbrains.gateway.ssh.SshHostTunnelConnector
import com.jetbrains.gateway.thinClientLink.ThinClientHandle
import com.jetbrains.rd.util.URI
import com.jetbrains.rd.util.concurrentMapOf
import com.jetbrains.rd.util.lifetime.Lifetime
import io.gitpod.gitpodprotocol.api.entities.WorkspaceInstance
import io.gitpod.jetbrains.icons.GitpodIcons
Expand All @@ -49,7 +51,7 @@ import kotlin.coroutines.coroutineContext

@Suppress("UnstableApiUsage", "OPT_IN_USAGE")
class GitpodConnectionProvider : GatewayConnectionProvider {

private val activeConnections = concurrentMapOf<String, Boolean>()
private val gitpod = service<GitpodConnectionService>()

private val httpClient = HttpClient.newBuilder()
Expand All @@ -59,6 +61,7 @@ class GitpodConnectionProvider : GatewayConnectionProvider {
private val jacksonMapper = jacksonObjectMapper()
.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false)


override suspend fun connect(
parameters: Map<String, String>,
requestor: ConnectionRequestor
Expand All @@ -74,6 +77,7 @@ class GitpodConnectionProvider : GatewayConnectionProvider {
parameters["workspaceId"]!!,
parameters["backendPort"]
)

val client = gitpod.obtainClient(connectParams.gitpodHost)
val connectionLifetime = Lifetime.Eternal.createNested()
val updates = client.listenToWorkspace(connectionLifetime, connectParams.workspaceId)
Expand Down Expand Up @@ -144,6 +148,22 @@ class GitpodConnectionProvider : GatewayConnectionProvider {
connectionPanel.repaint()
}

val connectionKeyId = "${connectParams.gitpodHost}-${connectParams.workspaceId}-${connectParams.backendPort}"

if (activeConnections.containsKey(connectionKeyId)) {
val message = "You are trying to connect to a workspace that has a session already open"
val title = "Oops"
val okButton = Messages.getOkButton()
val options = arrayOf(okButton)
val defaultIndex = 0
val icon = Messages.getWarningIcon()
Messages.showDialog(message, title, options, defaultIndex, icon)

val errMessage = "A connection to the same workspace already exists: $connectionKeyId"
thisLogger().warn(errMessage)
throw IllegalStateException(errMessage)
}

GlobalScope.launch {
var thinClient: ThinClientHandle? = null
var thinClientJob: Job? = null
Expand Down Expand Up @@ -230,6 +250,10 @@ class GitpodConnectionProvider : GatewayConnectionProvider {
setErrorMessage("failed to fetch JetBrains Gateway Join Link.")
return@launch
}

// TODO: is this the right place?
activeConnections[connectionKeyId] = true

val connector = ClientOverSshTunnelConnector(
connectionLifetime,
SshHostTunnelConnector(credentials),
Expand All @@ -239,6 +263,7 @@ class GitpodConnectionProvider : GatewayConnectionProvider {
clientHandle.clientClosed.advise(connectionLifetime) {
application.invokeLater {
connectionLifetime.terminate()
activeConnections.remove(connectionKeyId)
}
}
clientHandle.onClientPresenceChanged.advise(connectionLifetime) {
Expand Down

0 comments on commit 0b3d189

Please sign in to comment.