Skip to content

Commit

Permalink
TODO
Browse files Browse the repository at this point in the history
  • Loading branch information
mustard-mh committed Jul 15, 2024
1 parent fcba274 commit 140873a
Show file tree
Hide file tree
Showing 11 changed files with 125 additions and 71 deletions.
2 changes: 2 additions & 0 deletions .idea/misc.xml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

6 changes: 5 additions & 1 deletion components/ide/jetbrains/toolbox/build.gradle.kts
Original file line number Diff line number Diff line change
@@ -1,3 +1,7 @@
// Copyright (c) 2024 Gitpod GmbH. All rights reserved.
// Licensed under the GNU Affero General Public License (AGPL).
// See License.AGPL.txt in the project root for license information.

import com.github.jk1.license.filter.ExcludeTransitiveDependenciesFilter
import com.github.jk1.license.render.JsonReportRenderer
import org.jetbrains.intellij.pluginRepository.PluginRepositoryFactory
Expand Down Expand Up @@ -34,7 +38,7 @@ dependencies {
implementation("com.connectrpc:connect-kotlin:0.6.0")
// Java specific dependencies.
implementation("com.connectrpc:connect-kotlin-google-java-ext:0.6.0")
implementation("com.google.protobuf:protobuf-java:4.26.0")
implementation("com.google.protobuf:protobuf-java:4.27.2")
// WebSocket
compileOnly("javax.websocket:javax.websocket-api:1.1")
compileOnly("org.eclipse.jetty.websocket:websocket-api:9.4.54.v20240208")
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -60,19 +60,44 @@ class GitpodAuthManager {
manager.addEventListener {
when (it.type) {
AuthEvent.Type.LOGIN -> {
logger.info("account ${it.accountId} logged in")
logger.debug("account ${it.accountId} logged in")
resetCurrentAccount(it.accountId)
loginListeners.forEach { it() }
}

AuthEvent.Type.LOGOUT -> {
logger.info("account ${it.accountId} logged out")
logger.debug("account ${it.accountId} logged out")
resetCurrentAccount(it.accountId)
logoutListeners.forEach { it() }
}
}
}
}

private fun resetCurrentAccount(accountId: String) {
val account = manager.accountsWithStatus.find { it.account.id == accountId }?.account ?: return
logger.debug("reset settings for ${account.getHost()}")
Utils.gitpodSettings.resetSettings(account.getHost())
}

fun getCurrentAccount(): GitpodAccount? {
return manager.accountsWithStatus.firstOrNull()?.account
return manager.accountsWithStatus.find { it.account.getHost() === Utils.gitpodSettings.gitpodHost }?.account
}

fun loginWithHost(host: String): Boolean {
if (getCurrentAccount()?.getHost() == host) {
// TODO: validate token is still available
return true
}
val account = manager.accountsWithStatus.find { it.account.getHost() == host }?.account
if (account != null) {
Utils.gitpodSettings.gitpodHost = host
loginListeners.forEach { it() }
// TODO: validate token is still available
return true
}
Utils.openUrl(this.getOAuthLoginUrl(host))
return false
}

fun logout() {
Expand Down Expand Up @@ -135,48 +160,12 @@ class GitpodAccount(
private val name: String,
private val host: String
) : Account {
private val orgSelectedListeners: MutableList<(String) -> Unit> = mutableListOf()
private val logger = LoggerFactory.getLogger(javaClass)

override fun getId() = id
override fun getFullName() = name
fun getCredentials() = credentials
fun getHost() = host

private fun getStoreKey(key: String) = "USER:${id}:${key}"

var organizationId: String?
get() = Utils.settingStore[getStoreKey("ORG")]
set(value){
if (value == null) {
return
}
Utils.settingStore[getStoreKey("ORG")] = value
orgSelectedListeners.forEach { it(value) }
}

var preferEditor: String?
get() = Utils.settingStore[getStoreKey("EDITOR")]
set(value){
if (value == null) {
return
}
Utils.settingStore[getStoreKey("EDITOR")] = value
}

var preferWorkspaceClass: String?
get() = Utils.settingStore[getStoreKey("WS_CLS")]
set(value){
if (value == null) {
return
}
Utils.settingStore[getStoreKey("WS_CLS")] = value
}

fun onOrgSelected(listener: (String) -> Unit) {
orgSelectedListeners.add(listener)
}

fun encode(): String {
return Json.encodeToString(this)
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -29,11 +29,11 @@ class GitpodOrganizationPage(val authManager: GitpodAuthManager, val publicApi:
val options = mutableListOf<AutocompleteItem>()
options.addAll(organizations.map { org ->
MenuItem(org.name, null, null) {
authManager.getCurrentAccount()?.organizationId = org.id
Utils.gitpodSettings.organizationId = org.id
Utils.toolboxUi.hideUiPage(this)
}
})
val orgName = organizations.find { it.id == authManager.getCurrentAccount()?.organizationId }?.name ?: ""
val orgName = organizations.find { it.id == Utils.gitpodSettings.organizationId }?.name ?: ""
AutocompleteTextField("Organization", orgName, options, 1.0f) {
if (it.isNullOrEmpty()) {
ValidationResult.Invalid("Organization is required")
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,12 +16,12 @@ import io.gitpod.toolbox.auth.GitpodLoginPage
import io.gitpod.toolbox.auth.GitpodOrganizationPage
import io.gitpod.toolbox.components.GitpodIcon
import io.gitpod.toolbox.components.SimpleButton
import io.gitpod.toolbox.service.ConnectParams
import io.gitpod.toolbox.service.GitpodPublicApiManager
import io.gitpod.toolbox.service.Utils
import kotlinx.coroutines.launch
import org.slf4j.LoggerFactory
import java.net.URI
import java.net.URLEncoder

class GitpodRemoteProvider(
private val consumer: RemoteEnvironmentConsumer,
Expand All @@ -36,20 +36,25 @@ class GitpodRemoteProvider(
private val environmentMap = mutableMapOf<String, GitpodRemoteProviderEnvironment>()

private val openInToolboxUriHandler = GitpodOpenInToolboxUriHandler { connectParams ->
if (!authManger.loginWithHost(connectParams.gitpodHost)) {
// TODO: store connectParams locally so that user doesn't need another dashboard click?
return@GitpodOpenInToolboxUriHandler
}
Utils.toolboxUi.showPluginEnvironmentsPage()
setEnvironmentVisibility(connectParams.workspaceId)
setEnvironmentVisibility(connectParams)
}

// TODO: multiple host support
private fun setEnvironmentVisibility(workspaceId: String) {
private fun setEnvironmentVisibility(connectParams: ConnectParams) {
val workspaceId = connectParams.workspaceId
logger.info("setEnvironmentVisibility $workspaceId")
Utils.toolboxUi.showWindow()
val env = environmentMap[workspaceId]
if (env != null) {
env.markActive()
Utils.clientHelper.setAutoConnectOnEnvironmentReady(workspaceId, "GO-233.15026.17", "/workspace/empty")
Utils.clientHelper.setAutoConnectOnEnvironmentReady(connectParams.uniqueID, "GO-233.15026.17", "/workspace/empty")
} else {
GitpodRemoteProviderEnvironment(authManger, workspaceId, publicApi).apply {
GitpodRemoteProviderEnvironment(authManger, connectParams, publicApi).apply {
environmentMap[workspaceId] = this
this.markActive()
consumer.consumeEnvironments(listOf(this))
Expand All @@ -76,14 +81,6 @@ class GitpodRemoteProvider(
if (workspaces.isEmpty()) {
return@collect
}
// TODO: Remove me
workspaces.forEach{
val host = URLEncoder.encode("https://exp-migration.preview.gitpod-dev.com", "UTF-8")
val workspaceId = URLEncoder.encode(it.id, "UTF-8")
val debugWorkspace = "false"
val newUri = "jetbrains://gateway/io.gitpod.toolbox.gateway/open-in-toolbox?host=${host}&workspaceId=${workspaceId}&debugWorkspace=${debugWorkspace}"
logger.info("workspace ${it.id} $newUri")
}
consumer.consumeEnvironments(workspaces.map {
val env = environmentMap[it.id]
if (env != null) {
Expand All @@ -99,9 +96,9 @@ class GitpodRemoteProvider(
}

private fun startup() {
val account = authManger.getCurrentAccount() ?: return
authManger.getCurrentAccount() ?: return
publicApi.setup()
val orgId = account.organizationId
val orgId = Utils.gitpodSettings.organizationId
logger.info("user logged in, current selected org: $orgId")
if (orgId != null) {
Utils.dataManager.startWatchWorkspaces(publicApi)
Expand All @@ -111,14 +108,19 @@ class GitpodRemoteProvider(
Utils.toolboxUi.showUiPage(organizationPage)
}
}
authManger.getCurrentAccount()?.onOrgSelected {
Utils.dataManager.startWatchWorkspaces(publicApi)
Utils.gitpodSettings.onSettingsChanged { key, _ ->
when (key) {
GitpodSettings.SettingKey.ORGANIZATION_ID.name -> {
Utils.dataManager.startWatchWorkspaces(publicApi)
}
}
}
}

override fun getOverrideUiPage(): UiPage? {
logger.info("getOverrideUiPage")
authManger.getCurrentAccount() ?: return loginPage
val account = authManger.getCurrentAccount()
logger.info("get override ui page for ${account?.getHost()}")
account ?: return loginPage
return null
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@ import io.gitpod.publicapi.v1.WorkspaceOuterClass
import io.gitpod.publicapi.v1.WorkspaceOuterClass.WorkspacePhase
import io.gitpod.toolbox.auth.GitpodAuthManager
import io.gitpod.toolbox.components.SimpleButton
import io.gitpod.toolbox.service.ConnectParams
import io.gitpod.toolbox.service.GitpodPublicApiManager
import io.gitpod.toolbox.service.Utils
import kotlinx.coroutines.channels.BufferOverflow
Expand All @@ -24,15 +25,15 @@ import java.util.concurrent.CompletableFuture

class GitpodRemoteProviderEnvironment(
private val authManager: GitpodAuthManager,
private val workspaceId: String,
private val connectParams: ConnectParams,
private val publicApi: GitpodPublicApiManager,
) : AbstractRemoteProviderEnvironment() {
private val logger = LoggerFactory.getLogger(javaClass)
private val actionList = Utils.observablePropertiesFactory.emptyObservableList<ActionDescription>();
private val contentsViewFuture: CompletableFuture<EnvironmentContentsView> = CompletableFuture.completedFuture(
GitpodSSHEnvironmentContentsView(
authManager,
workspaceId,
connectParams,
publicApi,
)
)
Expand All @@ -53,9 +54,9 @@ class GitpodRemoteProviderEnvironment(
}

init {
logger.info("==================GitpodRemoteProviderEnvironment.init $workspaceId")
logger.info("==================GitpodRemoteProviderEnvironment.init ${connectParams.uniqueID}")
Utils.coroutineScope.launch {
Utils.dataManager.watchWorkspaceStatus(workspaceId) {
Utils.dataManager.watchWorkspaceStatus(connectParams.workspaceId) {
lastPhase = it.phase
lastWSEnvState.tryEmit(WorkspaceEnvState(it.phase, isMarkActive))
}
Expand Down Expand Up @@ -88,8 +89,8 @@ class GitpodRemoteProviderEnvironment(
}
}

override fun getId(): String = workspaceId
override fun getName(): String = workspaceId
override fun getId(): String = connectParams.uniqueID
override fun getName(): String = connectParams.resolvedWorkspaceId

override fun getContentsView(): CompletableFuture<EnvironmentContentsView> = contentsViewFuture

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import com.jetbrains.toolbox.gateway.environments.ManualEnvironmentContentsView
import com.jetbrains.toolbox.gateway.environments.SshEnvironmentContentsView
import com.jetbrains.toolbox.gateway.ssh.SshConnectionInfo
import io.gitpod.toolbox.auth.GitpodAuthManager
import io.gitpod.toolbox.service.ConnectParams
import io.gitpod.toolbox.service.GitpodConnectionProvider
import io.gitpod.toolbox.service.GitpodPublicApiManager
import io.gitpod.toolbox.service.Utils
Expand All @@ -17,7 +18,7 @@ import java.util.concurrent.CompletableFuture

class GitpodSSHEnvironmentContentsView(
private val authManager: GitpodAuthManager,
private val workspaceId: String,
private val connectParams: ConnectParams,
private val publicApi: GitpodPublicApiManager,
) : SshEnvironmentContentsView, ManualEnvironmentContentsView {
private var cancel = {}
Expand All @@ -27,7 +28,7 @@ class GitpodSSHEnvironmentContentsView(

override fun getConnectionInfo(): CompletableFuture<SshConnectionInfo> {
return Utils.coroutineScope.future {
val provider = GitpodConnectionProvider(authManager, workspaceId, publicApi)
val provider = GitpodConnectionProvider(authManager, connectParams, publicApi)
val (connInfo, cancel) = provider.connect()
this@GitpodSSHEnvironmentContentsView.cancel = cancel
return@future connInfo
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
// Copyright (c) 2024 Gitpod GmbH. All rights reserved.
// Licensed under the GNU Affero General Public License (AGPL).
// See License.AGPL.txt in the project root for license information.

package io.gitpod.toolbox.gateway

import io.gitpod.toolbox.service.Utils
import org.slf4j.LoggerFactory

class GitpodSettings {
private val logger = LoggerFactory.getLogger(javaClass)
private val settingsChangedListeners: MutableList<(String, String) -> Unit> = mutableListOf()

private fun getStoreKey(key: SettingKey) = "GITPOD_SETTINGS:${key.name}"

private fun updateSetting(key: SettingKey, value: String) {
logger.debug("updateSetting ${key.name}=$value")
Utils.settingStore[getStoreKey(key)] = value
settingsChangedListeners.forEach { it(key.name, value) }
}

fun onSettingsChanged(listener: (String, String) -> Unit) {
settingsChangedListeners.add(listener)
}

var organizationId: String?
get() {
val value = Utils.settingStore[getStoreKey(SettingKey.ORGANIZATION_ID)]
return if (value.isNullOrBlank()) null else value
}
set(value) {
updateSetting(SettingKey.ORGANIZATION_ID, value ?: "")
}

fun resetSettings(host: String = "gitpod.io") {
logger.info("=============reset for $host")
gitpodHost = host
organizationId = ""
}

var gitpodHost: String
get() = Utils.settingStore[getStoreKey(SettingKey.GITPOD_HOST)] ?: "gitpod.io"
set(value) {
updateSetting(SettingKey.GITPOD_HOST, value)
}

enum class SettingKey {
ORGANIZATION_ID,
GITPOD_HOST
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -13,12 +13,13 @@ import kotlinx.serialization.Serializable

class GitpodConnectionProvider(
private val authManager: GitpodAuthManager,
private val workspaceId: String,
private val connectParams: ConnectParams,
private val publicApi: GitpodPublicApiManager,
) {
private val activeConnections = ConcurrentHashMap<String, Boolean>()

suspend fun connect(): Pair<SshConnectionInfo, () -> Unit> {
val workspaceId = connectParams.workspaceId
val workspace = publicApi.getWorkspace(workspaceId).workspace
val ownerTokenResp = publicApi.getWorkspaceOwnerToken(workspaceId)
val account = authManager.getCurrentAccount() ?: throw Exception("No account found")
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -50,7 +50,7 @@ class GitpodPublicApiManager(private val authManger: GitpodAuthManager) {
}

private val orgId: String
get() = account?.organizationId ?: throw IllegalStateException("Organization not selected")
get() = Utils.gitpodSettings.organizationId ?: throw IllegalStateException("Organization not selected")

suspend fun listOrganizations(): List<OrganizationOuterClass.Organization> {
val organizationApi = organizationApi ?: throw IllegalStateException("No client")
Expand Down
Loading

0 comments on commit 140873a

Please sign in to comment.