From bcd6030842c8c82c88cf6350ffd31e6fa86a91de Mon Sep 17 00:00:00 2001 From: Charlie Drage Date: Tue, 20 Dec 2022 14:21:52 -0500 Subject: [PATCH] feat: add warnings to provider, add warn re. docker socket ### What does this PR do? * Reworks the plugin and extension API's to provide the ability to pass "up" warnings about the provider * Adds a warning check for Podman to see if the docker socket is actually Podman in disguise and to warn the user ### Screenshot/screencast of this PR ![Screenshot 2022-12-15 at 3 51 18 PM](https://user-images.githubusercontent.com/6422176/207964858-5fb97a74-4f73-4952-8132-5b1af08ad572.png) ![Screenshot 2022-12-15 at 3 50 38 PM](https://user-images.githubusercontent.com/6422176/207964861-a9c1f72c-89d6-4816-beab-397af4125620.png) ### What issues does this PR fix or reference? Fixes https://github.com/containers/podman-desktop/issues/905 Fixes https://github.com/containers/podman-desktop/issues/758 ### How to test this PR? If you have podman and docker installed: 1. `yarn watch` 2. Start Podman Desktop 3. Start Docker Desktop (Podman Desktop should now warn you that the socket is not being used by Podman) 4. Quit Docker Desktop 5. `podman machine stop && podman machine start` or restart via UI on Podman Desktop in order for `podman machine` to re-enable using the /var/run/docker.sock 6. Warning should now clear If only using podman: 1. `yarn watch` 2. Use the podman provider 3. `sudo rm -f /var/run/docker.sock` or mv 4. Warning should appear that cannot find the docker socket Signed-off-by: Charlie Drage --- extensions/podman/src/extension.ts | 58 +++++++++- extensions/podman/src/warnings.ts | 101 ++++++++++++++++++ packages/extension-api/src/extension-api.d.ts | 14 +++ packages/main/src/plugin/api/provider-info.ts | 4 + packages/main/src/plugin/provider-impl.ts | 16 +++ packages/main/src/plugin/provider-registry.ts | 15 +++ packages/main/src/tray-menu.ts | 1 + .../src/lib/welcome/ProviderReady.svelte | 2 + .../src/lib/welcome/ProviderWarnings.svelte | 26 +++++ packages/renderer/src/stores/providers.ts | 3 + website/docs/troubleshooting.md | 34 ++++++ 11 files changed, 273 insertions(+), 1 deletion(-) create mode 100644 extensions/podman/src/warnings.ts create mode 100644 packages/renderer/src/lib/welcome/ProviderWarnings.svelte diff --git a/extensions/podman/src/extension.ts b/extensions/podman/src/extension.ts index c89f1f6effdf37..a1a2fdecf75240 100644 --- a/extensions/podman/src/extension.ts +++ b/extensions/podman/src/extension.ts @@ -29,6 +29,7 @@ import type { InstalledPodman } from './podman-cli'; import { execPromise, getPodmanCli, getPodmanInstallation } from './podman-cli'; import { PodmanConfiguration } from './podman-configuration'; import { getDetectionChecks } from './detection-checks'; +import { getDisguisedPodmanInformation, getSocketPath, isDisguisedPodman } from './warnings'; type StatusHandler = (name: string, event: extensionApi.ProviderConnectionStatus) => void; @@ -42,6 +43,11 @@ const podmanMachinesStatuses = new Map(); const currentConnections = new Map(); +// Warning to check to see if the socket is a disguised Podman socket, +// by default we assume it is until proven otherwise when we check +let isDisguisedPodmanSocket = true; +let disguisedPodmanSocketWatcher: extensionApi.FileSystemWatcher; + type MachineJSON = { Name: string; CPUs: number; @@ -343,6 +349,13 @@ export async function activate(extensionContext: extensionApi.ExtensionContext): ]; const provider = extensionApi.provider.createProvider(providerOptions); + + // Check on initial setup + checkDisguisedPodmanSocket(provider); + // update the status of the provider if the socket is changed, created or deleted + disguisedPodmanSocketWatcher = setupDisguisedPodmanSocketWatcher(provider, getSocketPath()); + extensionContext.subscriptions.push(disguisedPodmanSocketWatcher); + // provide an installation path ? if (podmanInstall.isAbleToInstall()) { provider.registerInstallation({ @@ -497,7 +510,7 @@ export async function activate(extensionContext: extensionApi.ExtensionContext): } // monitor provider - // like version, checks + // like version, checks, warnings monitorProvider(provider); // register the registries @@ -512,3 +525,46 @@ export function deactivate(): void { stopLoop = true; console.log('stopping podman extension'); } + +function setupDisguisedPodmanSocketWatcher( + provider: extensionApi.Provider, + socketFile: string, +): extensionApi.FileSystemWatcher { + // Monitor the socket file for any changes, creation or deletion + // and trigger a change if that happens + const socketWatcher = extensionApi.fs.createFileSystemWatcher(socketFile); + + console.log('socket file:', socketFile); + + socketWatcher.onDidChange(() => { + console.log('socket changed'); + checkDisguisedPodmanSocket(provider); + }); + + socketWatcher.onDidCreate(() => { + console.log('socket created'); + checkDisguisedPodmanSocket(provider); + }); + + socketWatcher.onDidDelete(() => { + console.log('socket deleted'); + checkDisguisedPodmanSocket(provider); + }); + + return socketWatcher; +} + +async function checkDisguisedPodmanSocket(provider: extensionApi.Provider) { + // Check to see if the socket is disguised or not. If it is, we'll push a warning up + // to the plugin library to the let the provider know that there is a warning + const disguisedCheck = await isDisguisedPodman(); + if (isDisguisedPodmanSocket != disguisedCheck) { + isDisguisedPodmanSocket = disguisedCheck; + } + + // If isDisguisedPodmanSocket is true, we'll push a warning up to the plugin library with getDisguisedPodmanWarning() + // If isDisguisedPodmanSocket is false, we'll push an empty array up to the plugin library to clear the warning + // as we have no other warnings to display (or implemented) + const retrievedWarnings = isDisguisedPodmanSocket ? [] : [getDisguisedPodmanInformation()]; + provider.updateWarning(retrievedWarnings); +} diff --git a/extensions/podman/src/warnings.ts b/extensions/podman/src/warnings.ts new file mode 100644 index 00000000000000..a0304b768e89a5 --- /dev/null +++ b/extensions/podman/src/warnings.ts @@ -0,0 +1,101 @@ +/********************************************************************** + * Copyright (C) 2022 Red Hat, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + * + * SPDX-License-Identifier: Apache-2.0 + ***********************************************************************/ + +import type * as extensionApi from '@tmpwip/extension-api'; +import * as os from 'node:os'; +import * as http from 'node:http'; + +// Explanations +const detailsExplanation = 'Podman is not emulating the default Docker socket path: '; +const detailsNotWorking = '. Docker-specific tools may not work.'; + +// Default socket paths +const windowsSocketPath = '//./pipe/docker_engine'; +const defaultSocketPath = '/var/run/docker.sock'; + +// Return the warning information to the user if the socket is not a disguised Podman socket +export function getDisguisedPodmanInformation(): extensionApi.ProviderInformation { + let details: string; + + // Set the details message based on the OS + switch (os.platform()) { + case 'win32': + details = detailsExplanation.concat(windowsSocketPath, detailsNotWorking); + break; + case 'darwin': + // Due to how `podman-mac-helper` does not (by default) map the emulator to /var/run/docker.sock, we need to explain + // that the user must go on the Podman Desktop website for more information. This is because the user must manually + // map the socket to /var/run/docker.sock if not done by `podman machine` already (podman machine automatically maps the socket if Docker is not running) + details = detailsExplanation.concat( + defaultSocketPath, + detailsNotWorking, + ' See troubleshooting page on podman-desktop.io for more information.', + ); + break; + default: + details = detailsExplanation.concat(defaultSocketPath, detailsNotWorking); + break; + } + + // Return ProviderInformation with the details message + return { + name: 'Docker Socket Compatibility', + details, + }; +} + +// Async function that checks to see if the current Docker socket is a disguised Podman socket +export async function isDisguisedPodman(): Promise { + const socketPath = getSocketPath(); + + const podmanPingUrl = { + path: '/libpod/_ping', + socketPath, + }; + + return new Promise(resolve => { + const req = http.get(podmanPingUrl, res => { + res.on('data', () => { + // do nothing + }); + + res.on('end', () => { + if (res.statusCode === 200) { + resolve(true); + } else { + resolve(false); + } + }); + }); + + req.once('error', err => { + console.debug('Error while pinging docker as podman', err); + resolve(false); + }); + }); +} + +// Function that checks whether you are running windows, mac or linux and returns back +// the correct Docker socket location +export function getSocketPath(): string { + let socketPath: string = defaultSocketPath; + if (os.platform() === 'win32') { + socketPath = windowsSocketPath; + } + return socketPath; +} diff --git a/packages/extension-api/src/extension-api.d.ts b/packages/extension-api/src/extension-api.d.ts index 1ea769db878e4e..fae577004e18c6 100644 --- a/packages/extension-api/src/extension-api.d.ts +++ b/packages/extension-api/src/extension-api.d.ts @@ -132,6 +132,13 @@ declare module '@tmpwip/extension-api' { status(): ProviderStatus; } + // For displaying essential information to the user + // "name" of the warning / title and a "details" field for more information + export interface ProviderInformation { + name: string; + details?: string; + } + export interface ProviderDetectionCheck { name: string; details?: string; @@ -146,6 +153,9 @@ declare module '@tmpwip/extension-api' { images?: ProviderImages; links?: ProviderLinks[]; detectionChecks?: ProviderDetectionCheck[]; + + // Provide way to add additional warnings to the provider + warning?: ProviderInformation[]; } export type ProviderConnectionStatus = 'started' | 'stopped' | 'starting' | 'stopping' | 'unknown'; @@ -295,6 +305,10 @@ declare module '@tmpwip/extension-api' { // it may happen after an update or an installation updateDetectionChecks(detectionChecks: ProviderDetectionCheck[]): void; + // update warning information for the provider + readonly warning: ProviderInformation[]; + updateWarning(warning: ProviderInformation[]): void; + // notify that detection checks have changed onDidUpdateDetectionChecks: Event; } diff --git a/packages/main/src/plugin/api/provider-info.ts b/packages/main/src/plugin/api/provider-info.ts index c02fd956b016c9..e4b16d11ae950e 100644 --- a/packages/main/src/plugin/api/provider-info.ts +++ b/packages/main/src/plugin/api/provider-info.ts @@ -23,6 +23,7 @@ import type { ProviderLinks, ProviderStatus, Link, + ProviderInformation, } from '@tmpwip/extension-api'; export type LifecycleMethod = 'start' | 'stop' | 'delete'; @@ -65,6 +66,9 @@ export interface ProviderInfo { links: ProviderLinks[]; detectionChecks: ProviderDetectionCheck[]; + // warning messages regarding the provider + warning: ProviderInformation[]; + images: ProviderImages; // can install a provider diff --git a/packages/main/src/plugin/provider-impl.ts b/packages/main/src/plugin/provider-impl.ts index ede246f34c32b2..94948a58ce49a9 100644 --- a/packages/main/src/plugin/provider-impl.ts +++ b/packages/main/src/plugin/provider-impl.ts @@ -37,6 +37,7 @@ import type { ProviderUpdate, ProviderAutostart, KubernetesProviderConnectionFactory, + ProviderInformation, } from '@tmpwip/extension-api'; import type { ProviderRegistry } from './provider-registry'; import { Emitter } from './events/emitter'; @@ -65,6 +66,10 @@ export class ProviderImpl implements Provider, IDisposable { private readonly _onDidUpdateDetectionChecks = new Emitter(); readonly onDidUpdateDetectionChecks: Event = this._onDidUpdateDetectionChecks.event; + private _warning: ProviderInformation[]; + private readonly _onDidUpdateWarning = new Emitter(); + readonly onDidUpdateWarning: Event = this._onDidUpdateWarning.event; + constructor( private _internalId: string, private providerOptions: ProviderOptions, @@ -80,6 +85,7 @@ export class ProviderImpl implements Provider, IDisposable { this._links = providerOptions.links || []; this._detectionChecks = providerOptions.detectionChecks || []; this._images = providerOptions.images || {}; + this._warning = providerOptions.warning || []; // monitor connection statuses setInterval(async () => { @@ -116,6 +122,10 @@ export class ProviderImpl implements Provider, IDisposable { return this._detectionChecks; } + get warning(): ProviderInformation[] { + return this._warning; + } + get images(): ProviderImages { return this._images; } @@ -139,6 +149,12 @@ export class ProviderImpl implements Provider, IDisposable { this._onDidUpdateDetectionChecks.fire(detectionChecks); } + // Update the warnings + updateWarning(warning: ProviderInformation[]): void { + this._warning = warning; + this._onDidUpdateWarning.fire(warning); + } + get status(): ProviderStatus { return this._status; } diff --git a/packages/main/src/plugin/provider-registry.ts b/packages/main/src/plugin/provider-registry.ts index cb7c6645e11013..54a763b215cb9c 100644 --- a/packages/main/src/plugin/provider-registry.ts +++ b/packages/main/src/plugin/provider-registry.ts @@ -111,18 +111,32 @@ export class ProviderRegistry { this.lifecycleListeners = []; this.containerConnectionLifecycleListeners = []; + // Every 2 seconds, we will check: + // * The status of the providers + // * Any new warnings or informations for each provider setInterval(async () => { Array.from(this.providers.keys()).forEach(providerKey => { + // Get the provider and its lifecycle const provider = this.providers.get(providerKey); const providerLifecycle = this.providerLifecycles.get(providerKey); + + // If the provider and its lifecycle exist, we will check if (provider && providerLifecycle) { + // Get the status const status = providerLifecycle.status(); + + // If the status does not match the current one, we will send a listener event and update the status if (status !== this.providerStatuses.get(providerKey)) { provider.updateStatus(status); this.listeners.forEach(listener => listener('provider:update-status', this.getProviderInfo(provider))); this.providerStatuses.set(providerKey, status); } } + + // Let the renderer know to check for new warnings and information + if (provider) { + this.apiSender.send('provider:update-warning', provider.id); + } }); }, 2000); } @@ -491,6 +505,7 @@ export class ProviderRegistry { detectionChecks: provider.detectionChecks, images: provider.images, version: provider.version, + warning: provider.warning, installationSupport, }; diff --git a/packages/main/src/tray-menu.ts b/packages/main/src/tray-menu.ts index 313328c4c6eba6..7d021dc32af145 100644 --- a/packages/main/src/tray-menu.ts +++ b/packages/main/src/tray-menu.ts @@ -78,6 +78,7 @@ export class TrayMenu { version: '', links: [], images: {}, + warning: [], installationSupport: false, containerProviderConnectionCreation: false, kubernetesProviderConnectionCreation: false, diff --git a/packages/renderer/src/lib/welcome/ProviderReady.svelte b/packages/renderer/src/lib/welcome/ProviderReady.svelte index dfc52148bce991..2f2440d6a63fe8 100644 --- a/packages/renderer/src/lib/welcome/ProviderReady.svelte +++ b/packages/renderer/src/lib/welcome/ProviderReady.svelte @@ -1,6 +1,7 @@ + +
+ + {#if providerInfo?.warning?.length > 0} + {#each providerInfo.warning as warn} +
+ ⚠️ + {warn.name}: + {warn.details} +
+ {/each} + {/if} +
diff --git a/packages/renderer/src/stores/providers.ts b/packages/renderer/src/stores/providers.ts index 1110fe317af6e4..a2d121dbe63f90 100644 --- a/packages/renderer/src/stores/providers.ts +++ b/packages/renderer/src/stores/providers.ts @@ -64,6 +64,9 @@ window?.events.receive('provider-delete', () => { window?.events.receive('provider:update-status', () => { fetchProviders(); }); +window?.events.receive('provider:update-warning', () => { + fetchProviders(); +}); window.addEventListener('system-ready', () => { fetchProviders(); }); diff --git a/website/docs/troubleshooting.md b/website/docs/troubleshooting.md index b7988e34b1b599..17256bfdd569b4 100644 --- a/website/docs/troubleshooting.md +++ b/website/docs/troubleshooting.md @@ -149,6 +149,40 @@ helper_binaries_dir=["/Users/user/example_directory"] **NOTE**: A pre-built binary will be added to the Podman release page so you do not have to build `podman-mac-helper`. An [issue is open for this](https://github.com/containers/podman/issues/16746). +### Warning about Docker compatibility mode + + +#### Issue: + +When running the Podman provider, a warning shows regarding Docker compatibility mode on the dashboard: + +```sh +⚠️ Docker Socket Compatibility: Podman is not emulating the default Docker socket path: '/var/run/docker.sock'. Docker-specific tools may not work. See troubleshooting page on podman-desktop.io for more information. +``` + +This may appear when either: +* The Docker socket is not mounted correctly +* Docker Desktop is also being ran at the same time + +#### Solution: + +**On macOS:** + +1. Stop Docker Desktop (if install) +2. Run the `podman-mac-helper` binary: +```sh +sudo podman-mac-helper install +``` +3. Restart the Podman machine (the default Docker socket path will be recreated and Podman will emulate it) + + +**On Linux / Windows:** + +1. Stop Docker Desktop (if installed) +2. Restart the Podman machine (the default Docker socket path will be recreated and Podman will emulate it) + +*Note:* If Docker Desktop is started again, it will automatically re-alias the default Docker socket location and the Podman compatibilty warning will re-appear. + ## Code Ready Containers