Skip to content

Commit

Permalink
feat: add launch VM button
Browse files Browse the repository at this point in the history
### What does this PR do?

* Using QEMU we add a feature to "Launch VM" in the background
* Uses websockets, qemu as well as our xterm.js library to achieve this
* Launches in "snapshot" mode so no data is written to .raw file so the
  file can be easily re-used

### Screenshot / video of UI

<!-- If this PR is changing UI, please include
screenshots or screencasts showing the difference -->

### What issues does this PR fix or reference?

<!-- Include any related issues from Podman Desktop
repository (or from another issue tracker). -->

Closes podman-desktop#813

### How to test this PR?

<!-- Please explain steps to reproduce -->

1. Be on macOS silicon
2. `brew install qemu`
3. Build a bootc container image
4. Press launch VM button in actions bar

Signed-off-by: Charlie Drage <[email protected]>
  • Loading branch information
cdrage committed Oct 17, 2024
1 parent d3e2e3e commit 376ff8f
Show file tree
Hide file tree
Showing 22 changed files with 710 additions and 11 deletions.
21 changes: 21 additions & 0 deletions docs/vm_launcher.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
# Virtual Machine Launcher

Virtual Machine support is **experimental** and is only meant to run *one VM at a time* within the BootC extension.

We launch the virtual machine by using QEMU.

There are some caveats however:
- The virtual machine is booted as a snapshot and writes data to a /tmp file. The .raw file will remain unmodified. All changes are discarded on shut down.
- VM is shutdown when changing to another page.
- Port 22 is forwarded to 2222 locally for SSH testing. The VM may be accessed by using ssh localhost -p 2222 on an external terminal.
- VM uses 4GB of RAM by default.

## Installation

### macOS

Install QEMU on macOS by running the following with `brew`:

```sh
brew install qemu
```
1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -65,6 +65,7 @@
"vitest": "^2.0.2"
},
"dependencies": {
"@xterm/addon-attach": "^0.11.0",
"js-yaml": "^4.1.0"
},
"packageManager": "[email protected]+sha512.73a29afa36a0d092ece5271de5177ecbf8318d454ecd701343131b8ebc0c1a91c487da46ab77c8e596d6acf1461e3594ced4becedf8921b074fbd8653ed7051c"
Expand Down
1 change: 1 addition & 0 deletions packages/backend/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -102,6 +102,7 @@
"@vitest/coverage-v8": "^2.0.2",
"@xterm/xterm": "^5.5.0",
"@xterm/addon-fit": "^0.10.0",
"@xterm/addon-attach": "^0.11.0",
"eslint": "^8.57.1",
"eslint-import-resolver-custom-alias": "^1.3.2",
"eslint-import-resolver-typescript": "^3.6.3",
Expand Down
41 changes: 40 additions & 1 deletion packages/backend/src/api-impl.ts
Original file line number Diff line number Diff line change
Expand Up @@ -25,10 +25,11 @@ import { History } from './history';
import * as containerUtils from './container-utils';
import { Messages } from '/@shared/src/messages/Messages';
import { telemetryLogger } from './extension';
import { checkPrereqs, isLinux, getUidGid } from './machine-utils';
import { checkPrereqs, isLinux, isMac, getUidGid } from './machine-utils';
import * as fs from 'node:fs';
import path from 'node:path';
import { getContainerEngine } from './container-utils';
import { checkVMLaunchPrereqs, launchVM, stopVM } from './launch-vm';
import examplesCatalog from '../assets/examples.json';
import type { ExamplesList } from '/@shared/src/models/examples';

Expand All @@ -52,6 +53,10 @@ export class BootcApiImpl implements BootcApi {
return checkPrereqs(await getContainerEngine());
}

async checkVMLaunchPrereqs(folder: string, architecture: string): Promise<string | undefined> {
return checkVMLaunchPrereqs(folder, architecture);
}

async buildExists(folder: string, types: BuildType[]): Promise<boolean> {
return buildExists(folder, types);
}
Expand All @@ -60,6 +65,31 @@ export class BootcApiImpl implements BootcApi {
return buildDiskImage(build, this.history, overwrite);
}

async launchVM(folder: string, architecture: string): Promise<void> {
try {
await launchVM(folder, architecture);
// Notify it has successfully launched
await this.notify(Messages.MSG_VM_LAUNCH_ERROR, { success: 'Launched!', error: '' });
} catch (e) {
// Make sure that we are able to display the "stderr" information if it exists as that actually shows
// the error when running the command.
let errorMessage: string;
if (e instanceof Error) {
// eslint-disable-next-line @typescript-eslint/no-explicit-any
errorMessage = `${e.message} ${'stderr' in e ? (e as any).stderr : ''}`;
} else {
errorMessage = String(e);
}
await this.notify(Messages.MSG_VM_LAUNCH_ERROR, { success: '', error: errorMessage });
}
return Promise.resolve();
}

// Stop VM by pid file on the system
async stopVM(): Promise<void> {
return stopVM();
}

async deleteBuilds(builds: BootcBuildInfo[]): Promise<void> {
const response = await podmanDesktopApi.window.showWarningMessage(
`Are you sure you want to remove the selected disk images from the build history? This will remove the history of the build as well as remove any lingering build containers.`,
Expand Down Expand Up @@ -253,6 +283,10 @@ export class BootcApiImpl implements BootcApi {
return isLinux();
}

async isMac(): Promise<boolean> {
return isMac();
}

async getUidGid(): Promise<string> {
return getUidGid();
}
Expand All @@ -278,6 +312,11 @@ export class BootcApiImpl implements BootcApi {
return undefined;
}

// Read from the podman desktop clipboard
async readFromClipboard(): Promise<string> {
return podmanDesktopApi.env.clipboard.readText();
}

// The API does not allow callbacks through the RPC, so instead
// we send "notify" messages to the frontend to trigger a refresh
// this method is internal and meant to be used by the API implementation
Expand Down
200 changes: 200 additions & 0 deletions packages/backend/src/launch-vm.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,200 @@
/**********************************************************************
* Copyright (C) 2024 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 path from 'node:path';
import * as extensionApi from '@podman-desktop/api';
import { isMac } from './machine-utils';
import fs from 'node:fs';

// Ignore the following line as this is where we will be storing the pid file
// similar to other projects that use pid files in /tmp
// eslint-disable-next-line sonarjs/publicly-writable-directories
const pidFile = '/tmp/qemu-podman-desktop.pid';

// Must use "homebrew" qemu binaries on macOS
// as they are found to be the most stable and reliable for the project
// as well as containing the necessary "edk2-aarch64-code.fd" file
// it is not advised to use the qemu binaries from qemu.org due to edk2-aarch64-code.fd not being included.
const macQemuArm64Binary = '/opt/homebrew/bin/qemu-system-aarch64';
const macQemuArm64Edk2 = '/opt/homebrew/share/qemu/edk2-aarch64-code.fd';
const macQemuX86Binary = '/opt/homebrew/bin/qemu-system-x86_64';

// Host port forwarding for VM we will by default port forward 22 on the bootable container
// to :2222 on the host
const hostForwarding = 'hostfwd=tcp::2222-:22';

// Default memory size for the VM and websocket port location
const memorySize = '4G';
const websocketPort = '45252';

// Raw image location
const rawImageLocation = 'image/disk.raw';

export async function launchVM(folder: string, architecture: string): Promise<void> {
// Will ONLY work with RAW images located at image/disk.raw which is the default output location
const diskImage = path.join(folder, rawImageLocation);

// Check to see that the disk image exists before continuing
if (!fs.existsSync(diskImage)) {
throw new Error(`Raw disk image not found: ${diskImage}`);
}

// Generate the launch command and then run process.exec
try {
const command = generateLaunchCommand(diskImage, architecture);

// If generateLaunchCommand returns an empty array, then we are not able to launch the VM
// so simply error out and return
if (command.length === 0) {
throw new Error(
'Unable to generate the launch command for the VM, must be on the appropriate OS (mac or linux) and architecture (x86_64 or aarch64)',
);
}

// Execute the command
await extensionApi.process.exec('sh', ['-c', `${command.join(' ')}`]);
} catch (e) {
// Output the stderr information if it exists as that helps with debugging
// why the command could not run.
if (e instanceof Error && 'stderr' in e) {
console.error('Error launching VM: ', e.stderr);
} else {
console.error('Error launching VM: ', e);
}
throw e;
}
}

// Stop VM by killing the process with the pid file (/tmp/qemu-podman-desktop.pid)
export async function stopVM(): Promise<void> {
try {
await extensionApi.process.exec('sh', ['-c', `kill -9 \`cat ${pidFile}\``]);
} catch (e) {
// If it errors out, we will ignore the error if it has stderr, is a string
// and contains 'No such process' as that means the process is not running
if (e instanceof Error && 'stderr' in e && typeof e.stderr === 'string' && e.stderr.includes('No such process')) {
return;
}

// if 'stderr' exists, we will throw an error with the stderr information
if (e instanceof Error && 'stderr' in e) {
throw new Error(typeof e.stderr === 'string' ? e.stderr : 'Unknown error');
} else {
throw new Error('Unknown error');
}
}
}

// Prereqs checks before launching the VM
export async function checkVMLaunchPrereqs(folder: string, architecture: string): Promise<string | undefined> {
// Check to see that the disk image exists before continuing
const diskImage = path.join(folder, rawImageLocation);
if (!fs.existsSync(diskImage)) {
return `Raw disk image not found at ${diskImage}. Please build a .raw disk image first.`;
}

// Check to see if the architecture is supported
if (architecture !== 'amd64' && architecture !== 'arm64') {
return `Unsupported architecture: ${architecture}`;
}

// If on macOS, check the qemu binaries exist as well as the edk2-aarch64-code.fd file
if (isMac()) {
const installDisclaimer = 'Please install qemu via our installation document';
if (!fs.existsSync(macQemuX86Binary)) {
return `QEMU x86 binary not found at ${macQemuX86Binary}. ${installDisclaimer}`;
}
if (architecture === 'arm64' && !fs.existsSync(macQemuArm64Binary)) {
return `QEMU arm64 binary not found at ${macQemuArm64Binary}. ${installDisclaimer}`;
}
if (architecture === 'arm64' && !fs.existsSync(macQemuArm64Edk2)) {
return `QEMU arm64 edk2-aarch64-code.fd file not found at ${macQemuArm64Edk2}. ${installDisclaimer}`;
}
}

return undefined;
}

// Generate launch command for qemu
// this all depends on what architecture we are launching as well as
// operating system
function generateLaunchCommand(diskImage: string, architecture: string): string[] {
let command: string[] = [];
switch (architecture) {
// Case for anything amd64
case 'amd64':
if (isMac()) {
command = [
macQemuX86Binary,
'-m',
memorySize,
'-nographic',
'-cpu',
'Broadwell-v4',
'-pidfile',
pidFile,
'-serial',
`websocket:127.0.0.1:${websocketPort},server,nowait`,
'-netdev',
`user,id=mynet0,${hostForwarding}`,
'-device',
'e1000,netdev=mynet0',
// Make sure we always have snapshot here as we don't want to modify the original image
'-snapshot',
diskImage,
];
}
break;

// For any arm64 images
case 'arm64':
if (isMac()) {
command = [
macQemuArm64Binary,
'-m',
memorySize,
'-nographic',
'-M',
'virt',
'-accel',
'hvf',
'-cpu',
'host',
'-smp',
'4',
'-serial',
`websocket:127.0.0.1:${websocketPort},server,nowait`,
'-pidfile',
pidFile,
'-netdev',
`user,id=usernet,${hostForwarding}`,
'-device',
'virtio-net,netdev=usernet',
'-drive',
`file=${macQemuArm64Edk2},format=raw,if=pflash,readonly=on`,
// Make sure we always have snapshot here as we don't want to modify the original image
'-snapshot',
diskImage,
];
}
break;
default:
break;
}
return command;
}
5 changes: 5 additions & 0 deletions packages/backend/src/machine-utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ import * as path from 'node:path';
import * as fs from 'node:fs';
import { satisfies, coerce } from 'semver';
import type { ContainerProviderConnection } from '@podman-desktop/api';
import { env } from '@podman-desktop/api';

function getMachineProviderEnv(connection: extensionApi.ContainerProviderConnection): string {
switch (connection.vmType) {
Expand Down Expand Up @@ -174,6 +175,10 @@ export function isLinux(): boolean {
return linux;
}

export function isMac(): boolean {
return env.isMac;
}

// Get the GID and UID of the current user and return in the format gid:uid
// in order for this to work, we must get this information from process.exec
// since there is no native way via node
Expand Down
1 change: 1 addition & 0 deletions packages/frontend/src/App.svelte
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import { onMount } from 'svelte';
import { getRouterState } from './api/client';
import { rpcBrowser } from '/@/api/client';
import { Messages } from '/@shared/src/messages/Messages';
import VM from './lib/disk-image/DiskImageDetailsVirtualMachine.svelte';
import DiskImageDetails from './lib/disk-image/DiskImageDetails.svelte';
import Navigation from './Navigation.svelte';
import DiskImagesList from './lib/disk-image/DiskImagesList.svelte';
Expand Down
3 changes: 3 additions & 0 deletions packages/frontend/src/Build.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -92,6 +92,7 @@ const mockImageInspect = {
} as unknown as ImageInspectInfo;

const mockIsLinux = false;
const mockIsMac = false;

vi.mock('./api/client', async () => {
return {
Expand All @@ -100,11 +101,13 @@ vi.mock('./api/client', async () => {
buildExists: vi.fn(),
listHistoryInfo: vi.fn(),
listBootcImages: vi.fn(),
listAllImages: vi.fn(),
inspectImage: vi.fn(),
inspectManifest: vi.fn(),
isLinux: vi.fn().mockImplementation(() => mockIsLinux),
generateUniqueBuildID: vi.fn(),
buildImage: vi.fn(),
isMac: vi.fn().mockImplementation(() => mockIsMac),
},
rpcBrowser: {
subscribe: () => {
Expand Down
16 changes: 16 additions & 0 deletions packages/frontend/src/VMConnectionStatus.svelte
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
<script lang="ts">
import Label from './lib/upstream/Label.svelte';
export let status: string;
function getClassColor(): string {
if (status.includes('VM stopped') || status.includes('VM error') || status.includes('VM launch error')) {
return 'bg-[var(--pd-status-disconnected)]';
}
return 'bg-[var(--pd-status-connected)]';
}
</script>

{#if status}
<Label role="status" name={status}><div class="w-2 h-2 {getClassColor()} rounded-full mx-1"></div></Label>
{/if}
1 change: 1 addition & 0 deletions packages/frontend/src/lib/dashboard/Dashboard.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -51,6 +51,7 @@ vi.mock('../../api/client', async () => {
listHistoryInfo: vi.fn(),
listBootcImages: vi.fn(),
pullImage: vi.fn(),
isMac: vi.fn(),
},
rpcBrowser: {
subscribe: () => {
Expand Down
Loading

0 comments on commit 376ff8f

Please sign in to comment.