שינוי של דיסק אחסון מתמיד

אתם יכולים להשתמש בדיסק אחסון מתמיד כדיסק אתחול למכונה וירטואלית (VM), או כדיסק נתונים שאתם מצרפים למכונה וירטואלית. במאמר הזה מוסבר איך לשנות נפחי אחסון קיימים של Persistent Disk כדי לבצע את הפעולות הבאות:

  • מעבר לסוג דיסק אחר.
  • מחיקה אוטומטית של דיסקים כשמכונות וירטואליות (VM) שמצורפות אליהם נמחקות.

מידע כללי על Persistent Disk זמין במאמר מידע על Persistent Disk.

שינוי הסוג של נפח האחסון של Persistent Disk

לפעמים צריך לשנות את הסוג של נפח מסוים של Persistent Disk כדי לעמוד בדרישות הביצועים או התמחור. לדוגמה, יכול להיות שתרצו לשנות את דיסק הנתונים של עומס עבודה מדיסק מתמיד סטנדרטי לדיסק אחסון מתמיד מאוזן.

אי אפשר לשנות ישירות את הסוג של נפח אחסון קיים של Persistent Disk. צריך ליצור snapshot של הדיסק הקיים ואז להשתמש ב-snapshot הזה כדי ליצור דיסק מהסוג החדש.

לדוגמה, כדי לשנות דיסק מתמיד סטנדרטי לדיסק מתמיד שמבוסס על SSD, משתמשים בתהליך הבא:

המסוף

  1. יוצרים קובץ snapshot של דיסק מתמיד סטנדרטי.
  2. יוצרים דיסק אחסון מתמיד חדש על סמך ה-snapshot. ברשימה הנפתחת סוג בוחרים באפשרות 'SSD persistent disk' (דיסק מתמיד שמבוסס על SSD).

gcloud

  1. יוצרים קובץ snapshot של דיסק מתמיד סטנדרטי.
  2. יוצרים דיסק אחסון מתמיד חדש על סמך ה-snapshot. כוללים את הדגל --type ומציינים את הערך pd-ssd.

REST

  1. יוצרים קובץ snapshot של דיסק מתמיד סטנדרטי.
  2. יוצרים דיסק אחסון מתמיד חדש על סמך ה-snapshot. בשדה type, מציינים את הערך "zones/ZONE/diskTypes/pd-ssd" ומחליפים את ZONE בתחום שבו נמצאים המכונה והדיסק החדש.

אחרי שיוצרים את הדיסק החדש ובודקים אותו, אפשר למחוק את ה-snapshot ולמחוק את הדיסק המקורי.

הגדרת מצב המחיקה האוטומטית של נפח אחסון של Persistent Disk

אתם יכולים למחוק באופן אוטומטי את אמצעי האחסון של Persistent Disk עם הרשאות קריאה/כתיבה כשמוחקים את מופע ה-VM שמשויך אליהם. ההתנהגות הזו נשלטת על ידי המאפיין autoDelete במופע של מכונה וירטואלית עבור דיסק מצורף נתון, ואפשר לעדכן אותה בכל שלב. באופן דומה, אפשר למנוע מחיקה של נפח של Persistent Disk על ידי סימון הערך autoDelete כ-false.

המסוף

  1. נכנסים לדף VM instances במסוף Google Cloud .

    כניסה לדף VM instances

  2. בוחרים את המופע שהדיסקים משויכים אליו.

  3. לוחצים על שם המכונה. יופיע הדף VM instance details.

  4. לוחצים על Edit.

  5. בקטע אחסון, מתחת לכותרת דיסקים נוספים, לוחצים על סמל העיפרון כדי לשנות את כלל המחיקה של הדיסק.

  6. לוחצים על שמירה כדי לעדכן את המופע.

gcloud

מגדירים את מצב המחיקה האוטומטית של Persistent Disk באמצעות הפקודה gcloud compute instances set-disk-auto-delete. כדי לשמור את הדיסק, משתמשים בדגל --no-auto-delete. כדי למחוק את הדיסק, משתמשים בדגל --auto-delete.

gcloud compute instances set-disk-auto-delete VM_NAME \
  AUTO_DELETE_SETTING \
  --disk DISK_NAME

מחליפים את מה שכתוב בשדות הבאים:

  • VM_NAME: השם של המכונה
  • AUTO_DELETE_SETTING: האם למחוק את הדיסק באופן אוטומטי. מציינים --no-auto-delete כדי לשמור את הדיסק אחרי מחיקת המכונה הווירטואלית, ו---auto-delete כדי למחוק את הדיסק בו-זמנית עם המכונה הווירטואלית.
  • DISK_NAME: שם הדיסק

Go

לפני שמנסים את הדוגמה הזו, צריך לפעול לפי Goההוראות להגדרה שבמדריך למתחילים של Compute Engine באמצעות ספריות לקוח. מידע נוסף מופיע במאמרי העזרה של Compute Engine Go API.

כדי לבצע אימות ב-Compute Engine, צריך להגדיר את Application Default Credentials. מידע נוסף זמין במאמר הגדרת אימות לסביבת פיתוח מקומית.

import (
	"context"
	"fmt"
	"io"

	compute "cloud.google.com/go/compute/apiv1"
	computepb "cloud.google.com/go/compute/apiv1/computepb"
)

// setDiskAutodelete sets the autodelete flag of a disk to given value.
func setDiskAutoDelete(
	w io.Writer,
	projectID, zone, instanceName, diskName string, autoDelete bool,
) error {
	// projectID := "your_project_id"
	// zone := "us-west3-b"
	// instanceName := "your_instance_name"
	// diskName := "your_disk_name"
	// autoDelete := true

	ctx := context.Background()
	instancesClient, err := compute.NewInstancesRESTClient(ctx)
	if err != nil {
		return fmt.Errorf("NewInstancesRESTClient: %w", err)
	}
	defer instancesClient.Close()

	getInstanceReq := &computepb.GetInstanceRequest{
		Project:  projectID,
		Zone:     zone,
		Instance: instanceName,
	}

	instance, err := instancesClient.Get(ctx, getInstanceReq)
	if err != nil {
		return fmt.Errorf("unable to get instance: %w", err)
	}

	diskExists := false

	for _, disk := range instance.GetDisks() {
		if disk.GetDeviceName() == diskName {
			diskExists = true
			break
		}
	}

	if !diskExists {
		return fmt.Errorf(
			"instance %s doesn't have a disk named %s attached",
			instanceName,
			diskName,
		)
	}

	req := &computepb.SetDiskAutoDeleteInstanceRequest{
		Project:    projectID,
		Zone:       zone,
		Instance:   instanceName,
		DeviceName: diskName,
		AutoDelete: autoDelete,
	}

	op, err := instancesClient.SetDiskAutoDelete(ctx, req)
	if err != nil {
		return fmt.Errorf("unable to set disk autodelete field: %w", err)
	}

	if err = op.Wait(ctx); err != nil {
		return fmt.Errorf("unable to wait for the operation: %w", err)
	}

	fmt.Fprintf(w, "disk autoDelete field updated.\n")

	return nil
}

Java

לפני שמנסים את הדוגמה הזו, צריך לפעול לפי Javaההוראות להגדרה שבמדריך למתחילים של Compute Engine באמצעות ספריות לקוח. מידע נוסף מופיע במאמרי העזרה של Compute Engine Java API.

כדי לבצע אימות ב-Compute Engine, צריך להגדיר את Application Default Credentials. מידע נוסף זמין במאמר הגדרת אימות לסביבת פיתוח מקומית.


import com.google.cloud.compute.v1.Instance;
import com.google.cloud.compute.v1.InstancesClient;
import com.google.cloud.compute.v1.Operation;
import com.google.cloud.compute.v1.SetDiskAutoDeleteInstanceRequest;
import java.io.IOException;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;

public class SetDiskAutodelete {

  public static void main(String[] args)
      throws IOException, ExecutionException, InterruptedException, TimeoutException {
    // TODO(developer): Replace these variables before running the sample.

    // Project ID or project number of the Cloud project you want to use.
    String projectId = "YOUR_PROJECT_ID";

    // The zone of the disk that you want to modify.
    String zone = "europe-central2-b";

    // Name of the instance the disk is attached to.
    String instanceName = "YOUR_INSTANCE_NAME";

    // The name of the disk for which you want to modify the autodelete flag.
    String diskName = "YOUR_DISK_NAME";

    // The new value of the autodelete flag.
    boolean autoDelete = true;

    setDiskAutodelete(projectId, zone, instanceName, diskName, autoDelete);
  }

  // Sets the autodelete flag of a disk to given value.
  public static void setDiskAutodelete(String projectId, String zone, String instanceName,
      String diskName, boolean autoDelete)
      throws IOException, ExecutionException, InterruptedException, TimeoutException {

    // Initialize client that will be used to send requests. This client only needs to be created
    // once, and can be reused for multiple requests. After completing all of your requests, call
    // the `instancesClient.close()` method on the client to safely
    // clean up any remaining background resources.
    try (InstancesClient instancesClient = InstancesClient.create()) {

      // Retrieve the instance given by the instanceName.
      Instance instance = instancesClient.get(projectId, zone, instanceName);

      // Check if the instance contains a disk that matches the given diskName.
      boolean diskNameMatch = instance.getDisksList()
          .stream()
          .anyMatch(disk -> disk.getDeviceName().equals(diskName));

      if (!diskNameMatch) {
        throw new Error(
            String.format("Instance %s doesn't have a disk named %s attached", instanceName,
                diskName));
      }

      // Create the request object.
      SetDiskAutoDeleteInstanceRequest request = SetDiskAutoDeleteInstanceRequest.newBuilder()
          .setProject(projectId)
          .setZone(zone)
          .setInstance(instanceName)
          .setDeviceName(diskName)
          // Update the autodelete property.
          .setAutoDelete(autoDelete)
          .build();

      // Wait for the update instance operation to complete.
      Operation response = instancesClient.setDiskAutoDeleteAsync(request)
          .get(3, TimeUnit.MINUTES);

      if (response.hasError()) {
        System.out.println("Failed to update Disk autodelete field!" + response);
        return;
      }
      System.out.println(
          "Disk autodelete field updated. Operation Status: " + response.getStatus());
    }
  }
}

Node.js

לפני שמנסים את הדוגמה הזו, צריך לפעול לפי Node.jsההוראות להגדרה שבמדריך למתחילים של Compute Engine באמצעות ספריות לקוח. מידע נוסף מופיע במאמרי העזרה של Compute Engine Node.js API.

כדי לבצע אימות ב-Compute Engine, צריך להגדיר את Application Default Credentials. מידע נוסף זמין במאמר הגדרת אימות לסביבת פיתוח מקומית.

/**
 * TODO(developer): Uncomment and replace these variables before running the sample.
 */
// const projectId = 'YOUR_PROJECT_ID';
// const zone = 'europe-central2-b';
// const instanceName = 'YOUR_INSTANCE_NAME';
// const diskName = 'YOUR_DISK_NAME';
// const autoDelete = true;

const compute = require('@google-cloud/compute');

async function setDiskAutodelete() {
  const instancesClient = new compute.InstancesClient();

  const [instance] = await instancesClient.get({
    project: projectId,
    zone,
    instance: instanceName,
  });

  if (!instance.disks.some(disk => disk.deviceName === diskName)) {
    throw new Error(
      `Instance ${instanceName} doesn't have a disk named ${diskName} attached.`
    );
  }

  const [response] = await instancesClient.setDiskAutoDelete({
    project: projectId,
    zone,
    instance: instanceName,
    deviceName: diskName,
    autoDelete,
  });
  let operation = response.latestResponse;
  const operationsClient = new compute.ZoneOperationsClient();

  // Wait for the update instance operation to complete.
  while (operation.status !== 'DONE') {
    [operation] = await operationsClient.wait({
      operation: operation.name,
      project: projectId,
      zone: operation.zone.split('/').pop(),
    });
  }

  console.log('Disk autoDelete field updated.');
}

setDiskAutodelete();

Python

לפני שמנסים את הדוגמה הזו, צריך לפעול לפי Pythonההוראות להגדרה שבמדריך למתחילים של Compute Engine באמצעות ספריות לקוח. מידע נוסף מופיע במאמרי העזרה של Compute Engine Python API.

כדי לבצע אימות ב-Compute Engine, צריך להגדיר את Application Default Credentials. מידע נוסף זמין במאמר הגדרת אימות לסביבת פיתוח מקומית.

from __future__ import annotations

import sys
from typing import Any

from google.api_core.extended_operation import ExtendedOperation
from google.cloud import compute_v1


def wait_for_extended_operation(
    operation: ExtendedOperation, verbose_name: str = "operation", timeout: int = 300
) -> Any:
    """
    Waits for the extended (long-running) operation to complete.

    If the operation is successful, it will return its result.
    If the operation ends with an error, an exception will be raised.
    If there were any warnings during the execution of the operation
    they will be printed to sys.stderr.

    Args:
        operation: a long-running operation you want to wait on.
        verbose_name: (optional) a more verbose name of the operation,
            used only during error and warning reporting.
        timeout: how long (in seconds) to wait for operation to finish.
            If None, wait indefinitely.

    Returns:
        Whatever the operation.result() returns.

    Raises:
        This method will raise the exception received from `operation.exception()`
        or RuntimeError if there is no exception set, but there is an `error_code`
        set for the `operation`.

        In case of an operation taking longer than `timeout` seconds to complete,
        a `concurrent.futures.TimeoutError` will be raised.
    """
    result = operation.result(timeout=timeout)

    if operation.error_code:
        print(
            f"Error during {verbose_name}: [Code: {operation.error_code}]: {operation.error_message}",
            file=sys.stderr,
            flush=True,
        )
        print(f"Operation ID: {operation.name}", file=sys.stderr, flush=True)
        raise operation.exception() or RuntimeError(operation.error_message)

    if operation.warnings:
        print(f"Warnings during {verbose_name}:\n", file=sys.stderr, flush=True)
        for warning in operation.warnings:
            print(f" - {warning.code}: {warning.message}", file=sys.stderr, flush=True)

    return result


def set_disk_autodelete(
    project_id: str, zone: str, instance_name: str, disk_name: str, autodelete: bool
) -> None:
    """
    Set the autodelete flag of a disk to given value.

    Args:
        project_id: project ID or project number of the Cloud project you want to use.
        zone: name of the zone in which is the disk you want to modify.
        instance_name: name of the instance the disk is attached to.
        disk_name: the name of the disk which flag you want to modify.
        autodelete: the new value of the autodelete flag.
    """
    instance_client = compute_v1.InstancesClient()
    instance = instance_client.get(
        project=project_id, zone=zone, instance=instance_name
    )

    for disk in instance.disks:
        if disk.device_name == disk_name:
            break
    else:
        raise RuntimeError(
            f"Instance {instance_name} doesn't have a disk named {disk_name} attached."
        )

    disk.auto_delete = autodelete

    operation = instance_client.update(
        project=project_id,
        zone=zone,
        instance=instance_name,
        instance_resource=instance,
    )

    wait_for_extended_operation(operation, "disk update")

REST

כדי להגדיר את מצב המחיקה האוטומטית באמצעות ה-API, שולחים בקשת POST אל ה-method‏ instances.setDiskAutoDelete.

משתמשים בפרמטר autoDelete כדי לציין אם למחוק את הדיסק.

POST https://compute.googleapis.com/compute/v1/projects/PROJECT_ID/zones/ZONE/instances/VM_NAME/setDiskAutoDelete?deviceName=DISK_NAME,autoDelete=AUTO_DELETE_OPTION

מחליפים את מה שכתוב בשדות הבאים:

  • PROJECT_ID: מזהה הפרויקט
  • ZONE: האזור שבו נמצאים המכונה והדיסק
  • VM_NAME: השם של המכונה
  • DISK_NAME: השם של הדיסק שמצורף למכונה.
  • AUTO_DELETE_OPTION: האם למחוק את הדיסק באופן אוטומטי כשהמכונה הווירטואלית נמחקת. כדי למחוק את הדיסק, מגדירים את הערך ל-true. מגדירים את הערך false כדי לשמור את הדיסק אחרי מחיקת המכונה הווירטואלית.

פתרון בעיות

כדי למצוא שיטות לאבחון ולפתרון בעיות שקשורות לדיסקים מלאים ולשינוי גודל הדיסק, ראו פתרון בעיות שקשורות לדיסקים מלאים ולשינוי גודל הדיסק.

המאמרים הבאים