Skip to content

Commit

Permalink
Merge pull request #2636 from digma-ai/persist-docker-compose
Browse files Browse the repository at this point in the history
Persist docker compose Closes #2621
  • Loading branch information
shalom938 authored Dec 29, 2024
2 parents c8a8bf7 + ad18beb commit 01535f6
Show file tree
Hide file tree
Showing 11 changed files with 456 additions and 202 deletions.
2 changes: 1 addition & 1 deletion common-build-logic/src/main/kotlin/common/BuildProfile.kt
Original file line number Diff line number Diff line change
Expand Up @@ -219,7 +219,7 @@ object BuildProfiles {
Profile.p243 to BuildProfile(
profile = Profile.p243,
platformVersion = "2024.3.1.1",
riderVersion = "2024.3.2",
riderVersion = "2024.3.3",
pycharmVersion = "2024.3",
riderTargetFramework = "net8.0",
riderResharperVersionConstant = "PROFILE_2023_2;PROFILE_2024_3",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@
import org.digma.intellij.plugin.analytics.BackendInfoHolder;
import org.digma.intellij.plugin.auth.account.*;
import org.digma.intellij.plugin.common.*;
import org.digma.intellij.plugin.docker.*;
import org.digma.intellij.plugin.errorreporting.ErrorReporter;
import org.digma.intellij.plugin.updates.ui.UIVersioningService;
import org.jetbrains.annotations.*;
Expand Down Expand Up @@ -382,7 +383,11 @@ private static JBLabel createBackendVersionLabel() {
if (someProject != null) {
var about = BackendInfoHolder.getInstance(someProject).getAbout();
if (about != null) {
backendVersionLabel.setText(about.getApplicationVersion());
if (LocalInstallationFacade.getInstance().isLocalEngineInstalled()) {
backendVersionLabel.setText(about.getApplicationVersion() + " (" + DockerService.getInstance().getComposeFilePath() + ")");
}else{
backendVersionLabel.setText(about.getApplicationVersion());
}
}
}
return backendVersionLabel;
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,272 @@
package org.digma.intellij.plugin.docker

import com.intellij.openapi.diagnostic.Logger
import org.digma.intellij.plugin.common.Retries
import org.digma.intellij.plugin.errorreporting.ErrorReporter
import org.digma.intellij.plugin.log.Log
import org.digma.intellij.plugin.paths.DigmaPathManager
import java.io.File
import java.io.FileOutputStream
import java.net.HttpURLConnection
import java.net.URI
import java.net.URL
import java.nio.file.Files
import java.nio.file.StandardCopyOption
import kotlin.io.path.deleteIfExists

private const val COMPOSE_FILE_URL = "https://get.digma.ai/"
const val COMPOSE_FILE_NAME = "docker-compose.yml"
private const val RESOURCE_LOCATION = "docker-compose"
const val COMPOSE_FILE_DIR_NAME = "digma-docker"

/**
* ComposeFileProvider is responsible for providing the docker-compose.yml file.
* the latest docker-compose.yml is bundled with the plugin in build time.
* on first request for the file It will unpack the file from the resource to the local file system if it does not exist.
* after unpacking the file it will be available for running docker operations and will not be unpacked again, unless it was deleted somehow.
* the file is saved to a persistence folder on user's machine and should not be deleted by the user. its like an installation file
* of the plugin and usually users will not delete it.
* when upgrading the local engine the docker service will call downloadLatestComposeFile to download the latest compose file. the file will be downloaded
* and override the existing file, from now on the new file will be used for docker operations.
* if after upgrade the file is deleted somehow, the plugin will use the bundled file from the resource again. this may cause a downgrade of the local
* engine, but we don't expect that to happen. (TODO: an improvement to this case can be to track upgrades, mark a flag in persistence that there was
* an upgrade and if the file is deleted after upgrade, download the latest again. this may also cause an issue if the latest backend is not
* compatible with this plugin version or the docker images are different then what is currently running).
* This class also supports using a custom docker compose file for development purposes only. the custom file is downloaded from a custom url provided
* with a system property. engine upgrade will not work when using a custom docker compose file.
*/
class ComposeFileProvider {

private val logger = Logger.getInstance(this::class.java)

private val composeFile: File = File(COMPOSE_FILE_DIR, COMPOSE_FILE_NAME)

//customComposeFileDownloaded is used to mark if the custom compose file was already downloaded in this IDE session.
// it needs to be downloaded only once per IDE session. after IDE restart, a new one will be downloaded and override the old one if it exists.
// it may be that a new session has another custom compose file url so need to override the old one.
private var customComposeFileDownloaded = false
private val customComposeFile: File = File(CUSTOM_COMPOSE_FILE_DIR, COMPOSE_FILE_NAME)


companion object {
val COMPOSE_FILE_DIR = File(DigmaPathManager.getLocalFilesDirectoryPath(), COMPOSE_FILE_DIR_NAME)
val CUSTOM_COMPOSE_FILE_DIR = File(System.getProperty("java.io.tmpdir"), COMPOSE_FILE_DIR_NAME)

private fun usingCustomComposeFile(): Boolean {
return getCustomComposeFileUrl() != null
}

/*
Using a custom compose file is for development purposes only and not for users.
After using it, it is necessary to remove the local engine and remove the property.
Then install local engine regularly.
Upgrade should not be invoked when using a custom compose file. it will not work, it will always use the same custom url.
*/
private fun getCustomComposeFileUrl(): String? {
return System.getProperty("org.digma.plugin.custom.docker-compose.url")
}
}


//this method should not be used to get the file itself, it is mainly for logging and debugging.
//it should not create the file or unpack it.
//use the method getComposeFile() to get the file for running docker operations
fun getComposeFilePath(): String {
if(usingCustomComposeFile()) {
return customComposeFile.absolutePath
}
return composeFile.absolutePath
}



//this method should return a file, if the file does not exist, the docker operation will fail
fun getComposeFile(): File {
ensureComposeFileExists()

if (usingCustomComposeFile()) {
return customComposeFile
}

return composeFile
}


fun ensureComposeFileExists(): Boolean {
try {
if (usingCustomComposeFile()) {
Log.log(logger::info, "using custom compose file {}", getCustomComposeFileUrl())
return ensureCustomComposeFileExists()
}

if (composeFile.exists()) {
Log.log(logger::info, "compose file exists {}", composeFile)
return true
}

Log.log(logger::info, "compose file does not exist, unpacking bundled file")
unpack()
return true

} catch (e: Throwable) {
Log.warnWithException(logger, e, "could not ensure compose file exists")
ErrorReporter.getInstance().reportError("ComposeFileProvider.ensureComposeFileExists", e)
return false
}
}


private fun ensureCustomComposeFileExists(): Boolean {

if (customComposeFileDownloaded && customComposeFile.exists()) {
return true
}

return getCustomComposeFileUrl()?.let { url ->
CUSTOM_COMPOSE_FILE_DIR.mkdirs()
downloadAndCopyFile(URI(url).toURL(), customComposeFile)
customComposeFileDownloaded = customComposeFile.exists()
customComposeFile.exists()
} ?: false

}



private fun unpack() {
Log.log(logger::info, "unpacking docker-compose.yml")

try {
ensureDirectoryExist()

if (COMPOSE_FILE_DIR.exists()) {
copyComposeFileFromResource()
Log.log(logger::info, "docker-compose.yml unpacked to {}", COMPOSE_FILE_DIR)
}
} catch (e: Exception) {
ErrorReporter.getInstance().reportError("ComposeFileProvider.unpack", e)
Log.warnWithException(logger, e, "could not unpack docker-compose.yml")
}
}


private fun ensureDirectoryExist() {
if (!COMPOSE_FILE_DIR.exists()) {
if (!COMPOSE_FILE_DIR.mkdirs()) {
Log.log(logger::warn, "could not create directory for docker-compose.yml {}", COMPOSE_FILE_DIR)
ErrorReporter.getInstance().reportError(
null, "ComposeFileProvider.ensureDirectoryExist",
"ensureDirectoryExist,could not create directory for docker-compose.yml in $COMPOSE_FILE_DIR",
mapOf("error hint" to "could not create directory for docker-compose.yml in $COMPOSE_FILE_DIR")
)
}
}
}


private fun copyComposeFileFromResource() {
val resourceLocation = "/$RESOURCE_LOCATION/$COMPOSE_FILE_NAME"
val inputStream = this::class.java.getResourceAsStream(resourceLocation)
if (inputStream == null) {
Log.log(logger::warn, "could not find file in resource for {}", resourceLocation)
ErrorReporter.getInstance().reportError(
null, "ComposeFileProvider.copyFileFromResource",
"could not extract docker-compose.yml from resource", mapOf(
"resource location" to resourceLocation
)
)
return
}

FileOutputStream(composeFile).use {
Log.log(logger::info, "unpacking {} to {}", COMPOSE_FILE_NAME, composeFile)
com.intellij.openapi.util.io.StreamUtil.copy(inputStream, it)
}

}



//downloadLatestComposeFile is used only for upgrading the local engine
fun downloadLatestComposeFile(): Boolean {

try {
//try to delete the current file, don't fail if delete fails
deleteFile()
} catch (e: Throwable) {
Log.warnWithException(logger, e, "could not delete compose file")
ErrorReporter.getInstance().reportError("ComposeFileProvider.downloadLatestComposeFile", e)
}

try {
ensureDirectoryExist()
downloadAndCopyFile(URI(COMPOSE_FILE_URL).toURL(), composeFile)
return composeFile.exists()
} catch (e: Throwable) {
Log.warnWithException(logger, e, "could not download latest compose file")
ErrorReporter.getInstance().reportError("ComposeFileProvider.downloadLatestComposeFile", e)
return false
}
}



private fun downloadAndCopyFile(url: URL, toFile: File) {

val tempFile = kotlin.io.path.createTempFile("tempComposeFile", ".yml")

try {

Retries.simpleRetry({

Log.log(logger::info, "downloading {}", url)

val connection: HttpURLConnection = url.openConnection() as HttpURLConnection
connection.connectTimeout = 10000
connection.readTimeout = 5000
connection.requestMethod = "GET"
val responseCode: Int = connection.getResponseCode()

if (responseCode != HttpURLConnection.HTTP_OK) {
throw RuntimeException("could not download file from $url")
} else {
connection.inputStream.use {
Files.copy(it, tempFile, StandardCopyOption.REPLACE_EXISTING)
}

Log.log(logger::info, "copying downloaded file {} to {}", tempFile, toFile)
try {
Files.move(tempFile, toFile.toPath(), StandardCopyOption.REPLACE_EXISTING, StandardCopyOption.ATOMIC_MOVE)
} catch (e: Exception) {
//ATOMIC_MOVE is not always supported, so try again on exception
Files.move(tempFile, toFile.toPath(), StandardCopyOption.REPLACE_EXISTING)
}
}
}, Throwable::class.java, 5000, 3)

} catch (e: Exception) {
Log.log(logger::warn, "could not download file {}, {}", url, e)

ErrorReporter.getInstance().reportError(
"ComposeFileProvider.downloadAndCopyFile", e, mapOf(
"url" to url.toString(),
"toFile" to toFile.toString()
)
)

} finally {
tempFile.deleteIfExists()
}
}


fun deleteFile() {
Retries.simpleRetry({
val file = if(usingCustomComposeFile()) customComposeFile else composeFile
Files.deleteIfExists(file.toPath())
}, Throwable::class.java, 100, 5)
}



}
Original file line number Diff line number Diff line change
Expand Up @@ -74,7 +74,7 @@ private fun isLocalEngineRunning(): Boolean {
return false
}

val projectName = COMPOSE_FILE_DIR
val projectName = COMPOSE_FILE_DIR_NAME

val dockerCmd = getDockerCommand()

Expand Down Expand Up @@ -104,7 +104,7 @@ private fun isAnyEngineRunning(): Boolean {
return false
}

val projectName = COMPOSE_FILE_DIR
val projectName = COMPOSE_FILE_DIR_NAME

val dockerCmd = getDockerCommand()

Expand Down
Loading

0 comments on commit 01535f6

Please sign in to comment.