Merge pull request #53466 from davidz627/localSSDUUID
Automatic merge from submit-queue. If you want to cherry-pick this change to another branch, please follow the instructions <a href="https://github.com/kubernetes/community/blob/master/contributors/devel/cherry-picks.md">here</a>. Add options for mounting SCSI or NVMe local SSD though Block or Filesystem and do all of that with UUID Fixes: #51431 Mount SCSI local SSD by UUID in /mnt/disks/by-uuid/, also allows for users to request and mount NVMe disks. Both types of disks will be accessable either through block or filesystem To see code in progress for NVMe and block support see working branch: https://github.com/davidz627/kubernetes/tree/localExt
This commit is contained in:
		| @@ -668,6 +668,7 @@ ENABLE_PROMETHEUS_TO_SD: $(yaml-quote ${ENABLE_PROMETHEUS_TO_SD:-false}) | |||||||
| ENABLE_POD_PRIORITY: $(yaml-quote ${ENABLE_POD_PRIORITY:-}) | ENABLE_POD_PRIORITY: $(yaml-quote ${ENABLE_POD_PRIORITY:-}) | ||||||
| CONTAINER_RUNTIME: $(yaml-quote ${CONTAINER_RUNTIME:-}) | CONTAINER_RUNTIME: $(yaml-quote ${CONTAINER_RUNTIME:-}) | ||||||
| CONTAINER_RUNTIME_ENDPOINT: $(yaml-quote ${CONTAINER_RUNTIME_ENDPOINT:-}) | CONTAINER_RUNTIME_ENDPOINT: $(yaml-quote ${CONTAINER_RUNTIME_ENDPOINT:-}) | ||||||
|  | NODE_LOCAL_SSDS_EXT: $(yaml-quote ${NODE_LOCAL_SSDS_EXT:-}) | ||||||
| LOAD_IMAGE_COMMAND: $(yaml-quote ${LOAD_IMAGE_COMMAND:-}) | LOAD_IMAGE_COMMAND: $(yaml-quote ${LOAD_IMAGE_COMMAND:-}) | ||||||
| EOF | EOF | ||||||
|   if [ -n "${KUBELET_PORT:-}" ]; then |   if [ -n "${KUBELET_PORT:-}" ]; then | ||||||
|   | |||||||
| @@ -36,6 +36,11 @@ MASTER_ROOT_DISK_SIZE=${MASTER_ROOT_DISK_SIZE:-$(get-master-root-disk-size)} | |||||||
| NODE_DISK_TYPE=${NODE_DISK_TYPE:-pd-standard} | NODE_DISK_TYPE=${NODE_DISK_TYPE:-pd-standard} | ||||||
| NODE_DISK_SIZE=${NODE_DISK_SIZE:-100GB} | NODE_DISK_SIZE=${NODE_DISK_SIZE:-100GB} | ||||||
| NODE_LOCAL_SSDS=${NODE_LOCAL_SSDS:-0} | NODE_LOCAL_SSDS=${NODE_LOCAL_SSDS:-0} | ||||||
|  | # An extension to local SSDs allowing users to specify block/fs and SCSI/NVMe devices | ||||||
|  | # Format of this variable will be "#,scsi/nvme,block/fs" you can specify multiple | ||||||
|  | # configurations by seperating them by a semi-colon ex. "2,scsi,fs;1,nvme,block" | ||||||
|  | # is a request for 2 SCSI formatted and mounted SSDs and 1 NVMe block device SSD. | ||||||
|  | NODE_LOCAL_SSDS_EXT=${NODE_LOCAL_SSDS_EXT:-""} | ||||||
| # Accelerators to be attached to each node. Format "type=<accelerator-type>,count=<accelerator-count>" | # Accelerators to be attached to each node. Format "type=<accelerator-type>,count=<accelerator-count>" | ||||||
| # More information on available GPUs here - https://cloud.google.com/compute/docs/gpus/ | # More information on available GPUs here - https://cloud.google.com/compute/docs/gpus/ | ||||||
| NODE_ACCELERATORS=${NODE_ACCELERATORS:-""} | NODE_ACCELERATORS=${NODE_ACCELERATORS:-""} | ||||||
|   | |||||||
| @@ -25,6 +25,9 @@ set -o errexit | |||||||
| set -o nounset | set -o nounset | ||||||
| set -o pipefail | set -o pipefail | ||||||
|  |  | ||||||
|  | readonly UUID_MNT_PREFIX="/mnt/disks/by-uuid/google-local-ssds" | ||||||
|  | readonly UUID_BLOCK_PREFIX="/dev/disk/by-uuid/google-local-ssds" | ||||||
|  |  | ||||||
| function setup-os-params { | function setup-os-params { | ||||||
|   # Reset core_pattern. On GCI, the default core_pattern pipes the core dumps to |   # Reset core_pattern. On GCI, the default core_pattern pipes the core dumps to | ||||||
|   # /sbin/crash_reporter which is more restrictive in saving crash dumps. So for |   # /sbin/crash_reporter which is more restrictive in saving crash dumps. So for | ||||||
| @@ -85,11 +88,85 @@ function create-dirs { | |||||||
|   fi |   fi | ||||||
| } | } | ||||||
|  |  | ||||||
| # Formats the given device ($1) if needed and mounts it at given mount point | # Gets the total number of $(1) and $(2) type disks specified | ||||||
|  | # by the user in ${NODE_LOCAL_SSDS_EXT} | ||||||
|  | function get-local-disk-num() { | ||||||
|  |   local interface="${1}" | ||||||
|  |   local format="${2}" | ||||||
|  |  | ||||||
|  |   localdisknum=0 | ||||||
|  |   if [[ ! -z "${NODE_LOCAL_SSDS_EXT}" ]]; then | ||||||
|  |     IFS=";" read -r -a ssdgroups <<< "${NODE_LOCAL_SSDS_EXT}" | ||||||
|  |     for ssdgroup in "${ssdgroups[@]}"; do | ||||||
|  |       IFS="," read -r -a ssdopts <<< "${ssdgroup}" | ||||||
|  |       local opnum="${ssdopts[0]}" | ||||||
|  |       local opinterface="${ssdopts[1]}" | ||||||
|  |       local opformat="${ssdopts[2]}" | ||||||
|  |  | ||||||
|  |       if [[ "${opformat,,}" == "${format,,}" && "${opinterface,,}" == "${interface,,}" ]]; then | ||||||
|  |         localdisknum=$((localdisknum+opnum)) | ||||||
|  |       fi | ||||||
|  |     done | ||||||
|  |   fi | ||||||
|  | } | ||||||
|  |  | ||||||
|  | # Creates a symlink for a ($1) so that it may be used as block storage | ||||||
|  | function safe-block-symlink(){ | ||||||
|  |   local device="${1}" | ||||||
|  |   local symdir="${2}" | ||||||
|  |    | ||||||
|  |   mkdir -p "${symdir}" | ||||||
|  |  | ||||||
|  |   get-or-generate-uuid "${device}" | ||||||
|  |   local myuuid="${retuuid}" | ||||||
|  |  | ||||||
|  |   local sym="${symdir}/local-ssd-${myuuid}" | ||||||
|  |   # Do not "mkdir -p ${sym}" as that will cause unintended symlink behavior | ||||||
|  |   ln -s "${device}" "${sym}" | ||||||
|  |   echo "Created a symlink for SSD $ssd at ${sym}" | ||||||
|  |   chmod a+w "${sym}" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | # Gets a pregenerated UUID from ${ssdmap} if it exists, otherwise generates a new | ||||||
|  | # UUID and places it inside ${ssdmap} | ||||||
|  | function get-or-generate-uuid(){ | ||||||
|  |   local device="${1}" | ||||||
|  |  | ||||||
|  |   local ssdmap="/home/kubernetes/localssdmap.txt" | ||||||
|  |   echo "Generating or getting UUID from ${ssdmap}" | ||||||
|  |  | ||||||
|  |   if [[ ! -e "${ssdmap}" ]]; then | ||||||
|  |     touch "${ssdmap}" | ||||||
|  |     chmod +w "${ssdmap}" | ||||||
|  |   fi | ||||||
|  |  | ||||||
|  |   # each line of the ssdmap looks like "${device} persistent-uuid" | ||||||
|  |   if [[ ! -z $(grep ${device} ${ssdmap}) ]]; then | ||||||
|  |     #create symlink based on saved uuid | ||||||
|  |     local myuuid=$(grep ${device} ${ssdmap} | cut -d ' ' -f 2) | ||||||
|  |   else | ||||||
|  |     # generate new uuid and add it to the map | ||||||
|  |     local myuuid=$(uuidgen) | ||||||
|  |     if [[ ! ${?} -eq 0 ]]; then | ||||||
|  |       echo "Failed to generate valid UUID with uuidgen" >&2 | ||||||
|  |       exit 2 | ||||||
|  |     fi | ||||||
|  |     echo "${device} ${myuuid}" >> "${ssdmap}" | ||||||
|  |   fi | ||||||
|  |  | ||||||
|  |   if [[ -z "${myuuid}" ]]; then | ||||||
|  |     echo "Failed to get a uuid for device ${device} when symlinking." >&2 | ||||||
|  |     exit 2 | ||||||
|  |   fi | ||||||
|  |  | ||||||
|  |   retuuid="${myuuid}" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #Formats the given device ($1) if needed and mounts it at given mount point | ||||||
| # ($2). | # ($2). | ||||||
| function safe-format-and-mount() { | function safe-format-and-mount() { | ||||||
|   device=$1 |   local device="${1}" | ||||||
|   mountpoint=$2 |   local mountpoint="${2}" | ||||||
|  |  | ||||||
|   # Format only if the disk is not already formatted. |   # Format only if the disk is not already formatted. | ||||||
|   if ! tune2fs -l "${device}" ; then |   if ! tune2fs -l "${device}" ; then | ||||||
| @@ -102,18 +179,135 @@ function safe-format-and-mount() { | |||||||
|   mount -o discard,defaults "${device}" "${mountpoint}" |   mount -o discard,defaults "${device}" "${mountpoint}" | ||||||
| } | } | ||||||
|  |  | ||||||
| # Local ssds, if present, are mounted at /mnt/disks/ssdN. | # Gets a devices UUID and bind mounts the device to mount location in | ||||||
|  | # /mnt/disks/by-id/ | ||||||
|  | function unique-uuid-bind-mount(){ | ||||||
|  |   local mountpoint="${1}" | ||||||
|  |   local actual_device="${2}" | ||||||
|  |  | ||||||
|  |   # Trigger udev refresh so that newly formatted devices are propagated in by-uuid | ||||||
|  |   udevadm control --reload-rules | ||||||
|  |   udevadm trigger | ||||||
|  |   udevadm settle  | ||||||
|  |  | ||||||
|  |   # grep the exact match of actual device, prevents substring matching | ||||||
|  |   local myuuid=$(ls -l /dev/disk/by-uuid/ | grep "/${actual_device}$" | tr -s ' ' | cut -d ' ' -f 9) | ||||||
|  |   # myuuid should be the uuid of the device as found in /dev/disk/by-uuid/  | ||||||
|  |   if [[ -z "${myuuid}" ]]; then | ||||||
|  |     echo "Failed to get a uuid for device ${actual_device} when mounting." >&2 | ||||||
|  |     exit 2 | ||||||
|  |   fi | ||||||
|  |  | ||||||
|  |   # bindpoint should be the full path of the to-be-bound device | ||||||
|  |   local bindpoint="${UUID_MNT_PREFIX}-${interface}-fs/local-ssd-${myuuid}" | ||||||
|  |  | ||||||
|  |   safe-bind-mount "${mountpoint}" "${bindpoint}" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | # Bind mounts device at mountpoint to bindpoint | ||||||
|  | function safe-bind-mount(){ | ||||||
|  |   local mountpoint="${1}" | ||||||
|  |   local bindpoint="${2}" | ||||||
|  |  | ||||||
|  |   # Mount device to the mountpoint | ||||||
|  |   mkdir -p "${bindpoint}" | ||||||
|  |   echo "Binding '${mountpoint}' at '${bindpoint}'" | ||||||
|  |   mount --bind "${mountpoint}" "${bindpoint}" | ||||||
|  |   chmod a+w "${bindpoint}" | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Mounts, bindmounts, or symlinks depending on the interface and format | ||||||
|  | # of the incoming device | ||||||
|  | function mount-ext(){ | ||||||
|  |   local ssd="${1}" | ||||||
|  |   local devicenum="${2}" | ||||||
|  |   local interface="${3}" | ||||||
|  |   local format="${4}" | ||||||
|  |    | ||||||
|  |  | ||||||
|  |   if [[ -z "${devicenum}" ]]; then | ||||||
|  |     echo "Failed to get the local disk number for device ${ssd}" >&2 | ||||||
|  |     exit 2 | ||||||
|  |   fi | ||||||
|  |  | ||||||
|  |   # TODO: Handle partitioned disks. Right now this code just ignores partitions | ||||||
|  |   if [[ "${format}" == "fs" ]]; then | ||||||
|  |     if [[ "${interface}" == "scsi" ]]; then | ||||||
|  |       local actual_device=$(readlink -f "${ssd}" | cut -d '/' -f 3) | ||||||
|  |       # Error checking | ||||||
|  |       if [[ "${actual_device}" != sd* ]]; then | ||||||
|  |         echo "'actual_device' is not of the correct format. It must be the kernel name of the device, got ${actual_device} instead" >&2 | ||||||
|  |         exit 1 | ||||||
|  |       fi | ||||||
|  |       local mountpoint="/mnt/disks/ssd${devicenum}" | ||||||
|  |     else | ||||||
|  |       # This path is required because the existing Google images do not | ||||||
|  |       # expose NVMe devices in /dev/disk/by-id so we are using the /dev/nvme instead | ||||||
|  |       local actual_device=$(echo ${ssd} | cut -d '/' -f 3) | ||||||
|  |       # Error checking | ||||||
|  |       if [[ "${actual_device}" != nvme* ]]; then | ||||||
|  |         echo "'actual_device' is not of the correct format. It must be the kernel name of the device, got ${actual_device} instead" >&2 | ||||||
|  |         exit 1 | ||||||
|  |       fi | ||||||
|  |       local mountpoint="/mnt/disks/ssd-nvme${devicenum}" | ||||||
|  |     fi | ||||||
|  |  | ||||||
|  |     safe-format-and-mount "${ssd}" "${mountpoint}" | ||||||
|  |     # We only do the bindmount if users are using the new local ssd request method | ||||||
|  |     # see https://github.com/kubernetes/kubernetes/pull/53466#discussion_r146431894 | ||||||
|  |     if [[ ! -z "${NODE_LOCAL_SSDS_EXT}" ]]; then | ||||||
|  |       unique-uuid-bind-mount "${mountpoint}" "${actual_device}" | ||||||
|  |     fi | ||||||
|  |   elif [[ "${format}" == "block" ]]; then | ||||||
|  |     local symdir="${UUID_BLOCK_PREFIX}-${interface}-block" | ||||||
|  |     safe-block-symlink "${ssd}" "${symdir}" | ||||||
|  |   else | ||||||
|  |     echo "Disk format must be either fs or block, got ${format}" | ||||||
|  |   fi | ||||||
|  | } | ||||||
|  |  | ||||||
|  | # Local ssds, if present, are mounted or symlinked to their appropriate | ||||||
|  | # locations | ||||||
| function ensure-local-ssds() { | function ensure-local-ssds() { | ||||||
|  |   get-local-disk-num "scsi" "block" | ||||||
|  |   local scsiblocknum="${localdisknum}" | ||||||
|  |   local i=0 | ||||||
|   for ssd in /dev/disk/by-id/google-local-ssd-*; do |   for ssd in /dev/disk/by-id/google-local-ssd-*; do | ||||||
|     if [ -e "${ssd}" ]; then |     if [ -e "${ssd}" ]; then | ||||||
|       ssdnum=`echo ${ssd} | sed -e 's/\/dev\/disk\/by-id\/google-local-ssd-\([0-9]*\)/\1/'` |       local devicenum=`echo ${ssd} | sed -e 's/\/dev\/disk\/by-id\/google-local-ssd-\([0-9]*\)/\1/'` | ||||||
|       ssdmount="/mnt/disks/ssd${ssdnum}/" |       if [[ "${i}" -lt "${scsiblocknum}" ]]; then | ||||||
|       mkdir -p ${ssdmount} |         mount-ext "${ssd}" "${devicenum}" "scsi" "block" | ||||||
|       safe-format-and-mount "${ssd}" ${ssdmount} |  | ||||||
|       echo "Mounted local SSD $ssd at ${ssdmount}" |  | ||||||
|       chmod a+w ${ssdmount} |  | ||||||
|       else |       else | ||||||
|       echo "No local SSD disks found." |         # GKE does not set NODE_LOCAL_SSDS so all non-block devices | ||||||
|  |         # are assumed to be filesystem devices | ||||||
|  |         mount-ext "${ssd}" "${devicenum}" "scsi" "fs" | ||||||
|  |       fi | ||||||
|  |       i=$((i+1)) | ||||||
|  |     else | ||||||
|  |       echo "No local SCSI SSD disks found." | ||||||
|  |     fi | ||||||
|  |   done | ||||||
|  |  | ||||||
|  |   # The following mounts or symlinks NVMe devices | ||||||
|  |   get-local-disk-num "nvme" "block" | ||||||
|  |   local nvmeblocknum="${localdisknum}" | ||||||
|  |   local i=0 | ||||||
|  |   for ssd in /dev/nvme*; do | ||||||
|  |     if [ -e "${ssd}" ]; then | ||||||
|  |       # This workaround to find if the NVMe device is a disk is required because | ||||||
|  |       # the existing Google images does not expose NVMe devices in /dev/disk/by-id | ||||||
|  |       if [[ `udevadm info --query=property --name=${ssd} | grep DEVTYPE | sed "s/DEVTYPE=//"` == "disk" ]]; then | ||||||
|  |         local devicenum=`echo ${ssd} | sed -e 's/\/dev\/nvme0n\([0-9]*\)/\1/'` | ||||||
|  |         if [[ "${i}" -lt "${nvmeblocknum}" ]]; then | ||||||
|  |           mount-ext "${ssd}" "${devicenum}" "nvme" "block" | ||||||
|  |         else | ||||||
|  |           mount-ext "${ssd}" "${devicenum}" "nvme" "fs" | ||||||
|  |         fi | ||||||
|  |         i=$((i+1)) | ||||||
|  |       fi | ||||||
|  |     else | ||||||
|  |       echo "No local NVMe SSD disks found." | ||||||
|     fi |     fi | ||||||
|   done |   done | ||||||
| } | } | ||||||
|   | |||||||
| @@ -18,6 +18,8 @@ | |||||||
|  |  | ||||||
| # Use the config file specified in $KUBE_CONFIG_FILE, or default to | # Use the config file specified in $KUBE_CONFIG_FILE, or default to | ||||||
| # config-default.sh. | # config-default.sh. | ||||||
|  | readonly GCE_MAX_LOCAL_SSD=8 | ||||||
|  |  | ||||||
| KUBE_ROOT=$(dirname "${BASH_SOURCE}")/../.. | KUBE_ROOT=$(dirname "${BASH_SOURCE}")/../.. | ||||||
| source "${KUBE_ROOT}/cluster/gce/${KUBE_CONFIG_FILE-"config-default.sh"}" | source "${KUBE_ROOT}/cluster/gce/${KUBE_CONFIG_FILE-"config-default.sh"}" | ||||||
| source "${KUBE_ROOT}/cluster/common.sh" | source "${KUBE_ROOT}/cluster/common.sh" | ||||||
| @@ -37,6 +39,11 @@ else | |||||||
|   exit 1 |   exit 1 | ||||||
| fi | fi | ||||||
|  |  | ||||||
|  | if [[ ${NODE_LOCAL_SSDS} -ge 1 ]] && [[ ! -z ${NODE_LOCAL_SSDS_EXT} ]] ; then | ||||||
|  |   echo -e "${color_red}Local SSD: Only one of NODE_LOCAL_SSDS and NODE_LOCAL_SSDS_EXT can be specified at once${color_norm}" >&2 | ||||||
|  |   exit 2 | ||||||
|  | fi | ||||||
|  |  | ||||||
| if [[ "${MASTER_OS_DISTRIBUTION}" == "gci" ]]; then | if [[ "${MASTER_OS_DISTRIBUTION}" == "gci" ]]; then | ||||||
|     DEFAULT_GCI_PROJECT=google-containers |     DEFAULT_GCI_PROJECT=google-containers | ||||||
|     if [[ "${GCI_VERSION}" == "cos"* ]]; then |     if [[ "${GCI_VERSION}" == "cos"* ]]; then | ||||||
| @@ -546,6 +553,29 @@ function get-template-name-from-version() { | |||||||
|   echo "${NODE_INSTANCE_PREFIX}-template-${1}" | cut -c 1-63 | sed 's/[\.\+]/-/g;s/-*$//g' |   echo "${NODE_INSTANCE_PREFIX}-template-${1}" | cut -c 1-63 | sed 's/[\.\+]/-/g;s/-*$//g' | ||||||
| } | } | ||||||
|  |  | ||||||
|  | # validates the NODE_LOCAL_SSDS_EXT variable  | ||||||
|  | function validate-node-local-ssds-ext(){ | ||||||
|  |   ssdopts="${1}" | ||||||
|  |  | ||||||
|  |   if [[ -z "${ssdopts[0]}" || -z "${ssdopts[1]}" || -z "${ssdopts[2]}" ]]; then | ||||||
|  | 	  echo -e "${color_red}Local SSD: NODE_LOCAL_SSDS_EXT is malformed, found ${ssdopts[0]-_},${ssdopts[1]-_},${ssdopts[2]-_} ${color_norm}" >&2 | ||||||
|  |     exit 2 | ||||||
|  |   fi | ||||||
|  |   if [[ "${ssdopts[1]}" != "scsi" && "${ssdopts[1]}" != "nvme" ]]; then | ||||||
|  |     echo -e "${color_red}Local SSD: Interface must be scsi or nvme, found: ${ssdopts[1]} ${color_norm}" >&2 | ||||||
|  |     exit 2 | ||||||
|  |   fi | ||||||
|  |   if [[ "${ssdopts[2]}" != "fs" && "${ssdopts[2]}" != "block" ]]; then | ||||||
|  |     echo -e "${color_red}Local SSD: Filesystem type must be fs or block, found: ${ssdopts[2]} ${color_norm}"  >&2 | ||||||
|  |     exit 2 | ||||||
|  |   fi | ||||||
|  |   local_ssd_ext_count=$((local_ssd_ext_count+ssdopts[0])) | ||||||
|  |   if [[ "${local_ssd_ext_count}" -gt "${GCE_MAX_LOCAL_SSD}" || "${local_ssd_ext_count}" -lt 1 ]]; then | ||||||
|  |     echo -e "${color_red}Local SSD: Total number of local ssds must range from 1 to 8, found: ${local_ssd_ext_count} ${color_norm}" >&2 | ||||||
|  |     exit 2 | ||||||
|  |   fi | ||||||
|  | } | ||||||
|  |  | ||||||
| # Robustly try to create an instance template. | # Robustly try to create an instance template. | ||||||
| # $1: The name of the instance template. | # $1: The name of the instance template. | ||||||
| # $2: The scopes flag. | # $2: The scopes flag. | ||||||
| @@ -587,6 +617,19 @@ function create-node-template() { | |||||||
|   fi |   fi | ||||||
|  |  | ||||||
|   local local_ssds="" |   local local_ssds="" | ||||||
|  |   local_ssd_ext_count=0 | ||||||
|  |   if [[ ! -z ${NODE_LOCAL_SSDS_EXT-""} ]]; then | ||||||
|  |     IFS=";" read -r -a ssdgroups <<< ${NODE_LOCAL_SSDS_EXT} | ||||||
|  |     for ssdgroup in "${ssdgroups[@]}" | ||||||
|  |     do | ||||||
|  |       IFS="," read -r -a ssdopts <<< ${ssdgroup} | ||||||
|  |       validate-node-local-ssds-ext ${ssdopts} | ||||||
|  |       for i in $(seq ${ssdopts[0]}); do | ||||||
|  |         local_ssds="$local_ssds--local-ssd=interface=${ssdopts[1]} " | ||||||
|  |       done | ||||||
|  |     done | ||||||
|  |   fi | ||||||
|  |    | ||||||
|   if [[ ! -z ${NODE_LOCAL_SSDS+x} ]]; then |   if [[ ! -z ${NODE_LOCAL_SSDS+x} ]]; then | ||||||
|     # The NODE_LOCAL_SSDS check below fixes issue #49171 |     # The NODE_LOCAL_SSDS check below fixes issue #49171 | ||||||
|     # Some versions of seq will count down from 1 if "seq 0" is specified |     # Some versions of seq will count down from 1 if "seq 0" is specified | ||||||
| @@ -597,6 +640,7 @@ function create-node-template() { | |||||||
|     fi |     fi | ||||||
|   fi |   fi | ||||||
|    |    | ||||||
|  |  | ||||||
|   local network=$(make-gcloud-network-argument \ |   local network=$(make-gcloud-network-argument \ | ||||||
|     "${NETWORK_PROJECT}" \ |     "${NETWORK_PROJECT}" \ | ||||||
|     "${REGION}" \ |     "${REGION}" \ | ||||||
|   | |||||||
| @@ -21,6 +21,8 @@ import ( | |||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"path" | 	"path" | ||||||
| 	"path/filepath" | 	"path/filepath" | ||||||
|  | 	"strconv" | ||||||
|  | 	"strings" | ||||||
| 	"time" | 	"time" | ||||||
|  |  | ||||||
| 	. "github.com/onsi/ginkgo" | 	. "github.com/onsi/ginkgo" | ||||||
| @@ -54,8 +56,22 @@ const ( | |||||||
| 	DirectoryLocalVolumeType LocalVolumeType = "dir" | 	DirectoryLocalVolumeType LocalVolumeType = "dir" | ||||||
| 	// creates a tmpfs and mounts it | 	// creates a tmpfs and mounts it | ||||||
| 	TmpfsLocalVolumeType LocalVolumeType = "tmpfs" | 	TmpfsLocalVolumeType LocalVolumeType = "tmpfs" | ||||||
|  | 	// tests based on local ssd at /mnt/disks/by-uuid/ | ||||||
|  | 	GCELocalSSDVolumeType LocalVolumeType = "gce-localssd-scsi-fs" | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | var setupLocalVolumeMap = map[LocalVolumeType]func(*localTestConfig) *localTestVolume{ | ||||||
|  | 	GCELocalSSDVolumeType:    setupLocalVolumeGCELocalSSD, | ||||||
|  | 	TmpfsLocalVolumeType:     setupLocalVolumeTmpfs, | ||||||
|  | 	DirectoryLocalVolumeType: setupLocalVolumeDirectory, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var cleanupLocalVolumeMap = map[LocalVolumeType]func(*localTestConfig, *localTestVolume){ | ||||||
|  | 	GCELocalSSDVolumeType:    cleanupLocalVolumeGCELocalSSD, | ||||||
|  | 	TmpfsLocalVolumeType:     cleanupLocalVolumeTmpfs, | ||||||
|  | 	DirectoryLocalVolumeType: cleanupLocalVolumeDirectory, | ||||||
|  | } | ||||||
|  |  | ||||||
| type localTestVolume struct { | type localTestVolume struct { | ||||||
| 	// Node that the volume is on | 	// Node that the volume is on | ||||||
| 	node *v1.Node | 	node *v1.Node | ||||||
| @@ -199,33 +215,37 @@ var _ = SIGDescribe("PersistentVolumes-local [Feature:LocalPersistentVolumes] [S | |||||||
| 		}) | 		}) | ||||||
| 	}) | 	}) | ||||||
|  |  | ||||||
| 	LocalVolumeTypes := []LocalVolumeType{DirectoryLocalVolumeType, TmpfsLocalVolumeType} | 	LocalVolumeTypes := []LocalVolumeType{DirectoryLocalVolumeType, TmpfsLocalVolumeType, GCELocalSSDVolumeType} | ||||||
|  | 	for _, tempTestVolType := range LocalVolumeTypes { | ||||||
|  | 		// New variable required for gingko test closures | ||||||
|  | 		testVolType := tempTestVolType | ||||||
|  | 		ctxString := fmt.Sprintf("when using volume type %s", testVolType) | ||||||
|  | 		Context(ctxString, func() { | ||||||
|  | 			BeforeEach(func() { | ||||||
|  | 				if testVolType == GCELocalSSDVolumeType { | ||||||
|  | 					SkipUnlessLocalSSDExists("scsi", "fs", config.node0) | ||||||
|  | 				} | ||||||
|  | 			}) | ||||||
|  |  | ||||||
| 			Context("when two pods mount a local volume at the same time", func() { | 			Context("when two pods mount a local volume at the same time", func() { | ||||||
| 				It("should be able to write from pod1 and read from pod2", func() { | 				It("should be able to write from pod1 and read from pod2", func() { | ||||||
| 			for _, testVolType := range LocalVolumeTypes { |  | ||||||
| 					var testVol *localTestVolume | 					var testVol *localTestVolume | ||||||
| 				By(fmt.Sprintf("local-volume-type: %s", testVolType)) |  | ||||||
| 					testVol = setupLocalVolumePVCPV(config, testVolType) | 					testVol = setupLocalVolumePVCPV(config, testVolType) | ||||||
| 					twoPodsReadWriteTest(config, testVol) | 					twoPodsReadWriteTest(config, testVol) | ||||||
| 					cleanupLocalVolume(config, testVol) | 					cleanupLocalVolume(config, testVol) | ||||||
| 			} |  | ||||||
| 		}) |  | ||||||
| 				}) | 				}) | ||||||
|  |  | ||||||
|  | 			}) | ||||||
| 			Context("when two pods mount a local volume one after the other", func() { | 			Context("when two pods mount a local volume one after the other", func() { | ||||||
| 				It("should be able to write from pod1 and read from pod2", func() { | 				It("should be able to write from pod1 and read from pod2", func() { | ||||||
| 			for _, testVolType := range LocalVolumeTypes { |  | ||||||
| 					var testVol *localTestVolume | 					var testVol *localTestVolume | ||||||
| 				By(fmt.Sprintf("local-volume-type: %s", testVolType)) |  | ||||||
| 					testVol = setupLocalVolumePVCPV(config, testVolType) | 					testVol = setupLocalVolumePVCPV(config, testVolType) | ||||||
| 					twoPodsReadWriteSerialTest(config, testVol) | 					twoPodsReadWriteSerialTest(config, testVol) | ||||||
| 					cleanupLocalVolume(config, testVol) | 					cleanupLocalVolume(config, testVol) | ||||||
| 			} |  | ||||||
| 				}) | 				}) | ||||||
| 			}) | 			}) | ||||||
|  |  | ||||||
| 			Context("when pod using local volume with non-existant path", func() { | 			Context("when pod using local volume with non-existant path", func() { | ||||||
|  |  | ||||||
| 				ep := &eventPatterns{ | 				ep := &eventPatterns{ | ||||||
| 					reason:  "FailedMount", | 					reason:  "FailedMount", | ||||||
| 					pattern: make([]string, 2)} | 					pattern: make([]string, 2)} | ||||||
| @@ -233,8 +253,6 @@ var _ = SIGDescribe("PersistentVolumes-local [Feature:LocalPersistentVolumes] [S | |||||||
| 				ep.pattern = append(ep.pattern, "does not exist") | 				ep.pattern = append(ep.pattern, "does not exist") | ||||||
|  |  | ||||||
| 				It("should not be able to mount", func() { | 				It("should not be able to mount", func() { | ||||||
| 			for _, testVolType := range LocalVolumeTypes { |  | ||||||
| 				By(fmt.Sprintf("local-volume-type: %s", testVolType)) |  | ||||||
| 					testVol := &localTestVolume{ | 					testVol := &localTestVolume{ | ||||||
| 						node:            config.node0, | 						node:            config.node0, | ||||||
| 						hostDir:         "/non-existent/location/nowhere", | 						hostDir:         "/non-existent/location/nowhere", | ||||||
| @@ -245,10 +263,9 @@ var _ = SIGDescribe("PersistentVolumes-local [Feature:LocalPersistentVolumes] [S | |||||||
| 					pod, err := createLocalPod(config, testVol) | 					pod, err := createLocalPod(config, testVol) | ||||||
| 					Expect(err).To(HaveOccurred()) | 					Expect(err).To(HaveOccurred()) | ||||||
| 					checkPodEvents(config, pod.Name, ep) | 					checkPodEvents(config, pod.Name, ep) | ||||||
| 			} |  | ||||||
| 		}) |  | ||||||
| 				}) | 				}) | ||||||
|  |  | ||||||
|  | 			}) | ||||||
| 			Context("when pod's node is different from PV's NodeAffinity", func() { | 			Context("when pod's node is different from PV's NodeAffinity", func() { | ||||||
|  |  | ||||||
| 				BeforeEach(func() { | 				BeforeEach(func() { | ||||||
| @@ -262,21 +279,15 @@ var _ = SIGDescribe("PersistentVolumes-local [Feature:LocalPersistentVolumes] [S | |||||||
| 					pattern: make([]string, 2)} | 					pattern: make([]string, 2)} | ||||||
| 				ep.pattern = append(ep.pattern, "MatchNodeSelector") | 				ep.pattern = append(ep.pattern, "MatchNodeSelector") | ||||||
| 				ep.pattern = append(ep.pattern, "NoVolumeNodeConflict") | 				ep.pattern = append(ep.pattern, "NoVolumeNodeConflict") | ||||||
| 		for _, testVolType := range LocalVolumeTypes { |  | ||||||
|  |  | ||||||
| 				It("should not be able to mount due to different NodeAffinity", func() { | 				It("should not be able to mount due to different NodeAffinity", func() { | ||||||
|  |  | ||||||
| 					testPodWithNodeName(config, testVolType, ep, config.nodes.Items[1].Name, makeLocalPodWithNodeAffinity) | 					testPodWithNodeName(config, testVolType, ep, config.nodes.Items[1].Name, makeLocalPodWithNodeAffinity) | ||||||
| 				}) | 				}) | ||||||
|  |  | ||||||
| 				It("should not be able to mount due to different NodeSelector", func() { | 				It("should not be able to mount due to different NodeSelector", func() { | ||||||
|  |  | ||||||
| 					testPodWithNodeName(config, testVolType, ep, config.nodes.Items[1].Name, makeLocalPodWithNodeSelector) | 					testPodWithNodeName(config, testVolType, ep, config.nodes.Items[1].Name, makeLocalPodWithNodeSelector) | ||||||
| 				}) | 				}) | ||||||
|  |  | ||||||
| 		} |  | ||||||
| 			}) | 			}) | ||||||
|  |  | ||||||
| 			Context("when pod's node is different from PV's NodeName", func() { | 			Context("when pod's node is different from PV's NodeName", func() { | ||||||
|  |  | ||||||
| 				BeforeEach(func() { | 				BeforeEach(func() { | ||||||
| @@ -290,15 +301,15 @@ var _ = SIGDescribe("PersistentVolumes-local [Feature:LocalPersistentVolumes] [S | |||||||
| 					pattern: make([]string, 2)} | 					pattern: make([]string, 2)} | ||||||
| 				ep.pattern = append(ep.pattern, "NodeSelectorTerm") | 				ep.pattern = append(ep.pattern, "NodeSelectorTerm") | ||||||
| 				ep.pattern = append(ep.pattern, "Storage node affinity check failed") | 				ep.pattern = append(ep.pattern, "Storage node affinity check failed") | ||||||
| 		for _, testVolType := range LocalVolumeTypes { |  | ||||||
|  |  | ||||||
| 				It("should not be able to mount due to different NodeName", func() { | 				It("should not be able to mount due to different NodeName", func() { | ||||||
|  |  | ||||||
| 					testPodWithNodeName(config, testVolType, ep, config.nodes.Items[1].Name, makeLocalPodWithNodeName) | 					testPodWithNodeName(config, testVolType, ep, config.nodes.Items[1].Name, makeLocalPodWithNodeName) | ||||||
| 				}) | 				}) | ||||||
| 		} |  | ||||||
| 			}) | 			}) | ||||||
|  |  | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	Context("when using local volume provisioner", func() { | 	Context("when using local volume provisioner", func() { | ||||||
| 		var volumePath string | 		var volumePath string | ||||||
|  |  | ||||||
| @@ -362,7 +373,6 @@ type makeLocalPodWith func(config *localTestConfig, volume *localTestVolume, nod | |||||||
|  |  | ||||||
| func testPodWithNodeName(config *localTestConfig, testVolType LocalVolumeType, ep *eventPatterns, nodeName string, makeLocalPodFunc makeLocalPodWith) { | func testPodWithNodeName(config *localTestConfig, testVolType LocalVolumeType, ep *eventPatterns, nodeName string, makeLocalPodFunc makeLocalPodWith) { | ||||||
| 	var testVol *localTestVolume | 	var testVol *localTestVolume | ||||||
| 	By(fmt.Sprintf("local-volume-type: %s", testVolType)) |  | ||||||
| 	testVol = setupLocalVolumePVCPV(config, testVolType) | 	testVol = setupLocalVolumePVCPV(config, testVolType) | ||||||
|  |  | ||||||
| 	pod := makeLocalPodFunc(config, testVol, nodeName) | 	pod := makeLocalPodFunc(config, testVol, nodeName) | ||||||
| @@ -486,16 +496,7 @@ func podNodeName(config *localTestConfig, pod *v1.Pod) (string, error) { | |||||||
| 	return runtimePod.Spec.NodeName, runtimePodErr | 	return runtimePod.Spec.NodeName, runtimePodErr | ||||||
| } | } | ||||||
|  |  | ||||||
| // setupLocalVolume setups a directory to user for local PV | func setupWriteTestFile(hostDir string, config *localTestConfig, localVolumeType LocalVolumeType) *localTestVolume { | ||||||
| func setupLocalVolume(config *localTestConfig, localVolumeType LocalVolumeType) *localTestVolume { |  | ||||||
| 	testDirName := "local-volume-test-" + string(uuid.NewUUID()) |  | ||||||
| 	hostDir := filepath.Join(hostBase, testDirName) |  | ||||||
|  |  | ||||||
| 	if localVolumeType == TmpfsLocalVolumeType { |  | ||||||
| 		createAndMountTmpfsLocalVolume(config, hostDir) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// populate volume with testFile containing testFileContent |  | ||||||
| 	writeCmd, _ := createWriteAndReadCmds(hostDir, testFile, testFileContent) | 	writeCmd, _ := createWriteAndReadCmds(hostDir, testFile, testFileContent) | ||||||
| 	By(fmt.Sprintf("Creating local volume on node %q at path %q", config.node0.Name, hostDir)) | 	By(fmt.Sprintf("Creating local volume on node %q at path %q", config.node0.Name, hostDir)) | ||||||
| 	err := framework.IssueSSHCommand(writeCmd, framework.TestContext.Provider, config.node0) | 	err := framework.IssueSSHCommand(writeCmd, framework.TestContext.Provider, config.node0) | ||||||
| @@ -507,7 +508,30 @@ func setupLocalVolume(config *localTestConfig, localVolumeType LocalVolumeType) | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| // Deletes the PVC/PV, and launches a pod with hostpath volume to remove the test directory | func setupLocalVolumeTmpfs(config *localTestConfig) *localTestVolume { | ||||||
|  | 	testDirName := "local-volume-test-" + string(uuid.NewUUID()) | ||||||
|  | 	hostDir := filepath.Join(hostBase, testDirName) | ||||||
|  | 	createAndMountTmpfsLocalVolume(config, hostDir) | ||||||
|  | 	// populate volume with testFile containing testFileContent | ||||||
|  | 	return setupWriteTestFile(hostDir, config, TmpfsLocalVolumeType) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func setupLocalVolumeGCELocalSSD(config *localTestConfig) *localTestVolume { | ||||||
|  | 	res, err := framework.IssueSSHCommandWithResult("ls /mnt/disks/by-uuid/google-local-ssds-scsi-fs/", framework.TestContext.Provider, config.node0) | ||||||
|  | 	Expect(err).NotTo(HaveOccurred()) | ||||||
|  | 	dirName := strings.Fields(res.Stdout)[0] | ||||||
|  | 	hostDir := "/mnt/disks/by-uuid/google-local-ssds-scsi-fs/" + dirName | ||||||
|  | 	// populate volume with testFile containing testFileContent | ||||||
|  | 	return setupWriteTestFile(hostDir, config, GCELocalSSDVolumeType) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func setupLocalVolumeDirectory(config *localTestConfig) *localTestVolume { | ||||||
|  | 	testDirName := "local-volume-test-" + string(uuid.NewUUID()) | ||||||
|  | 	hostDir := filepath.Join(hostBase, testDirName) | ||||||
|  | 	// populate volume with testFile containing testFileContent | ||||||
|  | 	return setupWriteTestFile(hostDir, config, DirectoryLocalVolumeType) | ||||||
|  | } | ||||||
|  |  | ||||||
| func cleanupLocalVolume(config *localTestConfig, volume *localTestVolume) { | func cleanupLocalVolume(config *localTestConfig, volume *localTestVolume) { | ||||||
| 	if volume == nil { | 	if volume == nil { | ||||||
| 		return | 		return | ||||||
| @@ -519,10 +543,30 @@ func cleanupLocalVolume(config *localTestConfig, volume *localTestVolume) { | |||||||
| 		framework.Failf("Failed to delete PV and/or PVC: %v", utilerrors.NewAggregate(errs)) | 		framework.Failf("Failed to delete PV and/or PVC: %v", utilerrors.NewAggregate(errs)) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if volume.localVolumeType == TmpfsLocalVolumeType { | 	cleanup := cleanupLocalVolumeMap[volume.localVolumeType] | ||||||
| 		unmountTmpfsLocalVolume(config, volume.hostDir) | 	cleanup(config, volume) | ||||||
| 	} | } | ||||||
|  |  | ||||||
|  | // Deletes the PVC/PV, and launches a pod with hostpath volume to remove the test directory | ||||||
|  | func cleanupLocalVolumeGCELocalSSD(config *localTestConfig, volume *localTestVolume) { | ||||||
|  | 	By("Removing the test directory") | ||||||
|  | 	removeCmd := fmt.Sprintf("rm %s", volume.hostDir+"/"+testFile) | ||||||
|  | 	err := framework.IssueSSHCommand(removeCmd, framework.TestContext.Provider, config.node0) | ||||||
|  | 	Expect(err).NotTo(HaveOccurred()) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Deletes the PVC/PV, and launches a pod with hostpath volume to remove the test directory | ||||||
|  | func cleanupLocalVolumeTmpfs(config *localTestConfig, volume *localTestVolume) { | ||||||
|  | 	unmountTmpfsLocalVolume(config, volume.hostDir) | ||||||
|  |  | ||||||
|  | 	By("Removing the test directory") | ||||||
|  | 	removeCmd := fmt.Sprintf("rm -r %s", volume.hostDir) | ||||||
|  | 	err := framework.IssueSSHCommand(removeCmd, framework.TestContext.Provider, config.node0) | ||||||
|  | 	Expect(err).NotTo(HaveOccurred()) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Deletes the PVC/PV, and launches a pod with hostpath volume to remove the test directory | ||||||
|  | func cleanupLocalVolumeDirectory(config *localTestConfig, volume *localTestVolume) { | ||||||
| 	By("Removing the test directory") | 	By("Removing the test directory") | ||||||
| 	removeCmd := fmt.Sprintf("rm -r %s", volume.hostDir) | 	removeCmd := fmt.Sprintf("rm -r %s", volume.hostDir) | ||||||
| 	err := framework.IssueSSHCommand(removeCmd, framework.TestContext.Provider, config.node0) | 	err := framework.IssueSSHCommand(removeCmd, framework.TestContext.Provider, config.node0) | ||||||
| @@ -703,7 +747,9 @@ func podRWCmdExec(pod *v1.Pod, cmd string) string { | |||||||
| // and create local PVC and PV | // and create local PVC and PV | ||||||
| func setupLocalVolumePVCPV(config *localTestConfig, localVolumeType LocalVolumeType) *localTestVolume { | func setupLocalVolumePVCPV(config *localTestConfig, localVolumeType LocalVolumeType) *localTestVolume { | ||||||
| 	By("Initializing test volume") | 	By("Initializing test volume") | ||||||
| 	testVol := setupLocalVolume(config, localVolumeType) | 	setupLocalVolume, ok := setupLocalVolumeMap[localVolumeType] | ||||||
|  | 	Expect(ok).To(BeTrue()) | ||||||
|  | 	testVol := setupLocalVolume(config) | ||||||
|  |  | ||||||
| 	By("Creating local PVC and PV") | 	By("Creating local PVC and PV") | ||||||
| 	createLocalPVCPV(config, testVol) | 	createLocalPVCPV(config, testVol) | ||||||
| @@ -921,3 +967,16 @@ func findLocalPersistentVolume(c clientset.Interface, volumePath string) (*v1.Pe | |||||||
| 	} | 	} | ||||||
| 	return nil, fmt.Errorf("Unable to find local persistent volume with path %v", volumePath) | 	return nil, fmt.Errorf("Unable to find local persistent volume with path %v", volumePath) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // SkipUnlessLocalSSDExists takes in an ssdInterface (scsi/nvme) and a filesystemType (fs/block) | ||||||
|  | // and skips if a disk of that type does not exist on the node | ||||||
|  | func SkipUnlessLocalSSDExists(ssdInterface, filesystemType string, node *v1.Node) { | ||||||
|  | 	ssdCmd := fmt.Sprintf("ls -1 /mnt/disks/by-uuid/google-local-ssds-%s-%s/ | wc -l", ssdInterface, filesystemType) | ||||||
|  | 	res, err := framework.IssueSSHCommandWithResult(ssdCmd, framework.TestContext.Provider, node) | ||||||
|  | 	Expect(err).NotTo(HaveOccurred()) | ||||||
|  | 	num, err := strconv.Atoi(strings.TrimSpace(res.Stdout)) | ||||||
|  | 	Expect(err).NotTo(HaveOccurred()) | ||||||
|  | 	if num < 1 { | ||||||
|  | 		framework.Skipf("Requires at least 1 %s %s localSSD ", ssdInterface, filesystemType) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Kubernetes Submit Queue
					Kubernetes Submit Queue