#!/bin/bash

# -------------------------------------------------------------------------- #
# Copyright 2002-2025, OpenNebula Project, OpenNebula Systems                #
#                                                                            #
# 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.                                             #
#--------------------------------------------------------------------------- #

# clone fe:SOURCE host:remote_system_ds/disk.i size
#   - fe is the front-end hostname
#   - SOURCE is the path of the disk image in the form DS_BASE_PATH/disk
#   - host is the target host to deploy the VM
#   - remote_system_ds is the path for the system datastore in the host

SRC=$1
DST=$2
VM_ID=$3
DS_ID=$4

if [ -z "${ONE_LOCATION}" ]; then
    TMCOMMON=/var/lib/one/remotes/tm/tm_common.sh
else
    TMCOMMON=$ONE_LOCATION/var/remotes/tm/tm_common.sh
fi

DRIVER_PATH=$(dirname $0)

source $TMCOMMON
source ${DRIVER_PATH}/../../datastore/libfs.sh
source ${DRIVER_PATH}/../../etc/tm/fs_lvm/fs_lvm.conf
source ${DRIVER_PATH}/../../scripts_common.sh

#-------------------------------------------------------------------------------
# Set dst path and dir
#-------------------------------------------------------------------------------

DST_PATH=`arg_path $DST`
DST_HOST=`arg_host $DST`
DST_DIR=`dirname $DST_PATH`

SRC_ARG_PATH=`arg_path $SRC`

DST_DS_PATH="$(dirname $(dirname $(dirname $DST_PATH)))"
SRC_DS_PATH="$(dirname $(dirname $SRC_ARG_PATH))"

SRC_PATH="${DST_DS_PATH}${SRC_ARG_PATH##$SRC_DS_PATH}"

SRC_HOST=`arg_host $SRC`

DS_SYS_ID=$(echo $DST_DIR | $AWK -F '/' '{print $(NF-1)}')

disable_local_monitoring $DST_HOST $DST_DIR

#-------------------------------------------------------------------------------
# Get Image information
#-------------------------------------------------------------------------------

DISK_ID=$(basename ${DST_PATH} | cut -d. -f2)

XPATH="${DRIVER_PATH}/../../datastore/xpath.rb --stdin"

unset i j XPATH_ELEMENTS

while IFS= read -r -d '' element; do
    XPATH_ELEMENTS[i++]="$element"
done < <(onevm show -x $VM_ID| $XPATH \
                    /VM/TEMPLATE/DISK[DISK_ID=$DISK_ID]/SIZE \
                    /VM/TEMPLATE/DISK[DISK_ID=$DISK_ID]/ORIGINAL_SIZE \
                    /VM/TEMPLATE/DISK[DISK_ID=$DISK_ID]/LVM_THIN_ENABLE \
                    /VM/HISTORY_RECORDS/HISTORY[last\(\)]/TM_MAD)

SIZE="${XPATH_ELEMENTS[j++]}"
ORIGINAL_SIZE="${XPATH_ELEMENTS[j++]}"
LVM_THIN_ENABLE="${XPATH_ELEMENTS[j++],,}"
TM_MAD="${XPATH_ELEMENTS[j++]}"

if [ ! -r "${SRC_PATH}" ]; then
    log_error "Can't access image file '${SRC_PATH}' from frontend, please check datastore configuration"
    exit 1
fi

FILE_SIZE=`fs_size "${SRC_PATH}" YES`
if [ $? -ne 0 ]; then
    FILE_SIZE=0
fi

if [ $FILE_SIZE -gt $SIZE ]; then
    SIZE="$FILE_SIZE"
fi

FILE_TYPE=`file --brief ${SRC_PATH}`
# Extend the SIZE so the encrypted file fits in the LV
if [[ "$FILE_TYPE" =~ "LUKS encrypted file" ]]; then
    SIZE=$(( SIZE + 4 ))
fi

# Compute the seek start for zeroing of the new LV as the smaller
# number of ONE or file image virtual size. We start to zero from
# 1 MiB before the end of image size prior to the copying.
if [ "${FILE_SIZE}" -lt "${SIZE}" ]; then
    ZERO_SEEK_BYTES=$FILE_SIZE
else
    ZERO_SEEK_BYTES=$SIZE
fi

if [ "${ZERO_SEEK_BYTES}" -gt 0 ]; then
    ZERO_SEEK_BYTES=$(( (ZERO_SEEK_BYTES-1) * 1024 * 1024 ))
else
    ZERO_SEEK_BYTES=0
fi

#-------------------------------------------------------------------------------
# Create the snapshot and link it
#-------------------------------------------------------------------------------

POOL_NAME="lv-one-$VM_ID-pool"
LV_NAME="lv-one-$VM_ID-$DISK_ID"
VG_NAME="vg-one-$DS_SYS_ID"
DEV="/dev/${VG_NAME}/${LV_NAME}"

# Execute lvcreate with a lock in the frontend
CREATE_CMD=$(cat <<EOF
    set -e -o pipefail
    $SYNC
    $SUDO $LVSCAN
    if [ "$LVM_THIN_ENABLE" = 'yes' ]; then
        # If pool exists, extend it for the new disk. Otherwise, create it.
        if [ -n "\$($SUDO $LVS --noheading -S "vg_name = $VG_NAME && lv_name = $POOL_NAME")" ]; then
            $SUDO $LVEXTEND -L +${SIZE}M $VG_NAME/$POOL_NAME
        else
            # -ky is to enable activation skip, preventing possible data corruption for
            # example in HA situations
            # -Zy is for auto-zeroing thin volumes. It already defaults to 'y' but just in case...
            $SUDO $LVCREATE -ky -Zy -L ${SIZE}M --type thin-pool --thinpool $POOL_NAME $VG_NAME
            $SUDO $LVCHANGE -an $VG_NAME/$POOL_NAME
            $SYNC
        fi
        $SUDO $LVCREATE -ky --thin $VG_NAME/$POOL_NAME -V ${SIZE}M -n $LV_NAME
        $SUDO $LVCHANGE -ay -K $VG_NAME/$LV_NAME
    else
        $SUDO $LVCREATE --wipesignatures n -L${SIZE}M -n $LV_NAME $VG_NAME
    fi
    $SYNC
EOF
)

LOCK="tm-fs_lvm-${DS_SYS_ID}.lock"
exclusive "${LOCK}" 120 ssh_exec_and_log "$DST_HOST" "$CREATE_CMD" \
        "Error creating LV named $LV_NAME"

CLONE_CMD=$(cat <<EOF
    set -e -o pipefail
    mkdir -p $DST_DIR

    hostname -f >"${DST_DIR}/.host" || :

    # zero trailing space
    if [ "${ZERO_LVM_ON_CREATE}" = "yes" ] && [ "$LVM_THIN_ENABLE" != 'yes' ]; then
        LVSIZE=\$(${SUDO} ${LVS} --nosuffix --noheadings --units B -o lv_size "${DEV}" | tr -d '[:blank:]')
        ${DD} if=/dev/zero of="${DEV}" bs=${DD_BLOCK_SIZE:-64k} \
            oflag=seek_bytes iflag=count_bytes \
            seek="${ZERO_SEEK_BYTES}" count="\$(( LVSIZE - ${ZERO_SEEK_BYTES} ))"
    fi

    rm -f "$DST_PATH"
    ln -s "$DEV" "$DST_PATH"

    $SSH -n $SRC_HOST "$TAR -cSO $SRC_PATH 2> /dev/null" | $TAR -xO 2> /dev/null > "$DEV"
EOF
)

ssh_forward ssh_exec_and_log "$DST_HOST" "$CLONE_CMD" \
        "Error cloning $SRC_PATH to $LV_NAME"

exit 0
