| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535 |
- // Copyright 2019 Yunion
- //
- // 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.
- package storagehandler
- import (
- "context"
- "fmt"
- "net/http"
- "strings"
- "yunion.io/x/jsonutils"
- "yunion.io/x/log"
- "yunion.io/x/pkg/errors"
- "yunion.io/x/pkg/utils"
- "yunion.io/x/onecloud/pkg/apis"
- "yunion.io/x/onecloud/pkg/apis/compute"
- "yunion.io/x/onecloud/pkg/appsrv"
- "yunion.io/x/onecloud/pkg/hostman/hostutils"
- "yunion.io/x/onecloud/pkg/hostman/storageman"
- "yunion.io/x/onecloud/pkg/hostman/storageman/backupstorage"
- "yunion.io/x/onecloud/pkg/hostman/storageman/lvmutils"
- "yunion.io/x/onecloud/pkg/httperrors"
- "yunion.io/x/onecloud/pkg/mcclient/auth"
- modules "yunion.io/x/onecloud/pkg/mcclient/modules/compute"
- "yunion.io/x/onecloud/pkg/util/procutils"
- )
- var (
- storageKeyWords = []string{"storages"}
- storageActionFuncs = map[string]storageActionFunc{
- "attach": storageAttach,
- "detach": storageDetach,
- "update": storageUpdate,
- }
- )
- type storageActionFunc func(context.Context, jsonutils.JSONObject) (interface{}, error)
- func AddStorageHandler(prefix string, app *appsrv.Application) {
- for _, keyWords := range storageKeyWords {
- app.AddHandler("POST",
- fmt.Sprintf("%s/%s/<action>", prefix, keyWords),
- auth.Authenticate(storageActions))
- app.AddHandler("POST",
- fmt.Sprintf("%s/%s/<storageId>/delete-snapshots", prefix, keyWords),
- auth.Authenticate(storageDeleteSnapshots))
- app.AddHandler("POST",
- fmt.Sprintf("%s/%s/<storageId>/delete-snapshot", prefix, keyWords),
- auth.Authenticate(storageDeleteSnapshot))
- app.AddHandler("GET",
- fmt.Sprintf("%s/%s/is-mount-point", prefix, keyWords),
- auth.Authenticate(storageVerifyMountPoint))
- app.AddHandler("GET",
- fmt.Sprintf("%s/%s/is-local-mount-point", prefix, keyWords),
- auth.Authenticate(storageIsLocalMountPoint))
- app.AddHandler("GET",
- fmt.Sprintf("%s/%s/is-vg-exist", prefix, keyWords),
- auth.Authenticate(storageIsVgExist))
- app.AddHandler("POST",
- fmt.Sprintf("%s/%s/delete-backup", prefix, keyWords),
- auth.Authenticate(storageDeleteBackup))
- app.AddHandler("POST",
- fmt.Sprintf("%s/%s/sync-backup", prefix, keyWords),
- auth.Authenticate(storageSyncBackup))
- app.AddHandler("POST",
- fmt.Sprintf("%s/%s/pack-instance-backup", prefix, keyWords),
- auth.Authenticate(storagePackInstanceBackup))
- app.AddHandler("POST",
- fmt.Sprintf("%s/%s/unpack-instance-backup", prefix, keyWords),
- auth.Authenticate(storageUnpackInstanceBackup))
- app.AddHandler("POST",
- fmt.Sprintf("%s/%s/sync-backup-storage", prefix, keyWords),
- auth.Authenticate(storageSyncBackupStorage))
- app.AddHandler("POST",
- fmt.Sprintf("%s/%s/<storageId>/clean-recycle-diskfiles", prefix, keyWords),
- auth.Authenticate(storageCleanRecycleDiskfiles))
- }
- }
- func storageIsLocalMountPoint(ctx context.Context, w http.ResponseWriter, r *http.Request) {
- _, query, _ := appsrv.FetchEnv(ctx, w, r)
- mountPoint, err := query.GetString("mount_point")
- if err != nil {
- hostutils.Response(ctx, w, httperrors.NewMissingParameterError("mount_point"))
- return
- }
- fs, err := procutils.NewRemoteCommandAsFarAsPossible(
- "findmnt", "-n", "-o", "FSTYPE", "--target", mountPoint,
- ).Output()
- if err != nil {
- log.Errorf("failed get source of mountpoint %s: %s", mountPoint, err)
- hostutils.Response(ctx, w, httperrors.NewInternalServerError("failed get source of mountpoint %s: %s", mountPoint, err))
- return
- }
- fsStr := strings.TrimSpace(string(fs))
- log.Infof("check %s file system is %s", mountPoint, fsStr)
- if utils.IsInStringArray(fsStr, []string{"ext", "ext2", "ext3", "ext4", "xfs", "btrfs", "jfs", "reiserfs", "ntfs", "fat32", "exfat", "zfs"}) {
- // local file system
- appsrv.SendStruct(w, map[string]interface{}{"is_local_mount_point": true})
- } else {
- appsrv.SendStruct(w, map[string]interface{}{"is_local_mount_point": false})
- }
- }
- func storageIsVgExist(ctx context.Context, w http.ResponseWriter, r *http.Request) {
- _, query, _ := appsrv.FetchEnv(ctx, w, r)
- vgName, err := query.GetString("vg_name")
- if err != nil {
- hostutils.Response(ctx, w, httperrors.NewMissingParameterError("vg_name"))
- return
- }
- if err := lvmutils.VgDisplay(vgName); err != nil {
- log.Errorf("vg %s display failed %s", vgName, err)
- hostutils.Response(ctx, w, httperrors.NewInternalServerError("%s", err.Error()))
- return
- }
- hostutils.ResponseOk(ctx, w)
- }
- func storageVerifyMountPoint(ctx context.Context, w http.ResponseWriter, r *http.Request) {
- _, query, _ := appsrv.FetchEnv(ctx, w, r)
- mountPoint, err := query.GetString("mount_point")
- if err != nil {
- hostutils.Response(ctx, w, httperrors.NewMissingParameterError("mount_point"))
- return
- }
- output, err := procutils.NewRemoteCommandAsFarAsPossible("mountpoint", mountPoint).Output()
- if err == nil {
- appsrv.SendStruct(w, map[string]interface{}{"is_mount_point": true})
- } else {
- appsrv.SendStruct(w, map[string]interface{}{
- "is_mount_point": false,
- "error": string(output),
- })
- }
- }
- func storageActions(ctx context.Context, w http.ResponseWriter, r *http.Request) {
- params, _, body := appsrv.FetchEnv(ctx, w, r)
- var action = params["<action>"]
- if f, ok := storageActionFuncs[action]; !ok {
- hostutils.Response(ctx, w, httperrors.NewNotFoundError("Not found"))
- } else {
- res, err := f(ctx, body)
- if err != nil {
- hostutils.Response(ctx, w, err)
- } else if res != nil {
- hostutils.Response(ctx, w, res)
- } else {
- hostutils.ResponseOk(ctx, w)
- }
- }
- }
- func storageAttach(ctx context.Context, body jsonutils.JSONObject) (interface{}, error) {
- mountPoint, err := body.GetString("mount_point")
- if err != nil {
- return nil, httperrors.NewMissingParameterError("mount_point")
- }
- storageType, _ := body.GetString("storage_type")
- storage := storageman.GetManager().NewSharedStorageInstance(mountPoint, storageType)
- if storage == nil {
- return nil, httperrors.NewBadRequestError("'Not Support Storage[%s] mount_point: %s", storageType, mountPoint)
- }
- storagecacheId, _ := body.GetString("storagecache_id")
- imagecachePath, _ := body.GetString("imagecache_path")
- storageId, _ := body.GetString("storage_id")
- storageName, _ := body.GetString("name")
- storageConf, _ := body.Get("storage_conf")
- storage.SetStoragecacheId(storagecacheId)
- if err := storage.SetStorageInfo(storageId, storageName, storageConf); err != nil {
- return nil, err
- }
- /*err = storage.SyncStorageSize()
- if err != nil {
- return nil, errors.Wrapf(err, "SyncStorageSize")
- }*/
- resp, err := storage.SyncStorageInfo()
- if err != nil {
- return nil, err
- }
- storageman.GetManager().InitSharedStorageImageCache(storageType, storagecacheId, imagecachePath, storage)
- storageman.GetManager().Storages = append(storageman.GetManager().Storages, storage)
- return resp, nil
- }
- func storageDetach(ctx context.Context, body jsonutils.JSONObject) (interface{}, error) {
- info := struct {
- MountPoint string
- StorageId string
- Name string
- }{}
- err := body.Unmarshal(&info)
- if err != nil {
- return nil, errors.Wrapf(err, "body.Unmarshal")
- }
- if len(info.StorageId) == 0 {
- return nil, httperrors.NewMissingParameterError("storage_id")
- }
- storage := storageman.GetManager().GetStorage(info.StorageId)
- if storage != nil {
- if err := storage.Detach(); err != nil {
- log.Errorf("detach storage %s failed: %s", storage.GetPath(), err)
- }
- storageman.GetManager().Remove(storage)
- }
- return nil, nil
- }
- func storageUpdate(ctx context.Context, body jsonutils.JSONObject) (interface{}, error) {
- storageId, err := body.GetString("storage_id")
- if err != nil {
- return nil, httperrors.NewMissingParameterError("storage_id")
- }
- storageConf, err := body.Get("storage_conf")
- if err != nil {
- return nil, httperrors.NewMissingParameterError("storage_conf")
- }
- storage := storageman.GetManager().GetStorage(storageId)
- params := jsonutils.NewDict()
- params.Set("details", jsonutils.JSONTrue)
- ret, err := modules.Hoststorages.Get(hostutils.GetComputeSession(context.Background()),
- storageman.GetManager().GetHostId(), storageId, params)
- if err != nil {
- log.Errorln(err)
- return nil, err
- }
- if ret == nil || storage == nil {
- return nil, httperrors.NewNotFoundError("Storage %s not found", storageId)
- }
- storageName, _ := ret.GetString("storage")
- if err := storage.SetStorageInfo(storageId, storageName, storageConf); err != nil {
- return nil, err
- }
- mountPoint, _ := ret.GetString("mount_point")
- storage.SetPath(mountPoint)
- return nil, nil
- }
- func storageSyncBackup(ctx context.Context, w http.ResponseWriter, r *http.Request) {
- _, _, body := appsrv.FetchEnv(ctx, w, r)
- backupId, err := body.GetString("backup_id")
- if err != nil {
- hostutils.Response(ctx, w, httperrors.NewMissingParameterError("backup_id"))
- return
- }
- backupStorageId, err := body.GetString("backup_storage_id")
- if err != nil {
- hostutils.Response(ctx, w, httperrors.NewMissingParameterError("backup_storage_id"))
- return
- }
- backupStorageAccessInfo, err := body.Get("backup_storage_access_info")
- if err != nil {
- hostutils.Response(ctx, w, httperrors.NewMissingParameterError("backup_storage_access_info"))
- return
- }
- backupStorage, err := backupstorage.GetBackupStorage(backupStorageId, backupStorageAccessInfo.(*jsonutils.JSONDict))
- if err != nil {
- hostutils.Response(ctx, w, err)
- return
- }
- exist, reason, err := backupStorage.IsBackupExists(backupId)
- if err != nil {
- hostutils.Response(ctx, w, err)
- return
- }
- var (
- ret = jsonutils.NewDict()
- status string
- )
- if exist {
- status = compute.BACKUP_EXIST
- } else {
- if len(reason) == 0 {
- status = compute.BACKUP_NOT_EXIST
- } else {
- log.Errorf("fetch snapshot exist failed reason:%s", reason)
- status = compute.BACKUP_STATUS_UNKNOWN
- }
- }
- ret.Set("status", jsonutils.NewString(status))
- hostutils.Response(ctx, w, ret)
- }
- func storageSyncBackupStorage(ctx context.Context, w http.ResponseWriter, r *http.Request) {
- _, _, body := appsrv.FetchEnv(ctx, w, r)
- backupStorageId, err := body.GetString("backup_storage_id")
- if err != nil {
- hostutils.Response(ctx, w, httperrors.NewMissingParameterError("backup_storage_id"))
- return
- }
- backupStorageAccessInfo, err := body.Get("backup_storage_access_info")
- if err != nil {
- hostutils.Response(ctx, w, httperrors.NewMissingParameterError("backup_storage_access_info"))
- return
- }
- backupStorage, err := backupstorage.GetBackupStorage(backupStorageId, backupStorageAccessInfo.(*jsonutils.JSONDict))
- if err != nil {
- hostutils.Response(ctx, w, err)
- return
- }
- exist, reason, err := backupStorage.IsOnline()
- if err != nil {
- hostutils.Response(ctx, w, err)
- return
- }
- var (
- ret = jsonutils.NewDict()
- status string
- )
- if exist {
- status = compute.BACKUPSTORAGE_STATUS_ONLINE
- } else {
- status = compute.BACKUPSTORAGE_STATUS_OFFLINE
- }
- ret.Set("status", jsonutils.NewString(status))
- ret.Set("reason", jsonutils.NewString(reason))
- hostutils.Response(ctx, w, ret)
- }
- func storagePackInstanceBackup(ctx context.Context, w http.ResponseWriter, r *http.Request) {
- _, _, body := appsrv.FetchEnv(ctx, w, r)
- if !checkOptions(ctx, w, body, "package_name", "backup_ids", "backup_storage_id", "backup_storage_access_info", "metadata") {
- return
- }
- pb := storageman.SStoragePackInstanceBackup{}
- err := body.Unmarshal(&pb)
- if err != nil {
- hostutils.Response(ctx, w, httperrors.NewInputParameterError("%s", err.Error()))
- return
- }
- hostutils.DelayTask(ctx, packInstanceBackup, &pb)
- hostutils.ResponseOk(ctx, w)
- }
- func storageUnpackInstanceBackup(ctx context.Context, w http.ResponseWriter, r *http.Request) {
- _, _, body := appsrv.FetchEnv(ctx, w, r)
- if !checkOptions(ctx, w, body, "package_name", "backup_storage_id", "backup_storage_access_info") {
- return
- }
- pb := storageman.SStorageUnpackInstanceBackup{}
- err := body.Unmarshal(&pb)
- if err != nil {
- hostutils.Response(ctx, w, httperrors.NewInputParameterError("%s", err.Error()))
- return
- }
- hostutils.DelayTask(ctx, unpackInstanceBackup, &pb)
- hostutils.ResponseOk(ctx, w)
- }
- func packInstanceBackup(ctx context.Context, params interface{}) (jsonutils.JSONObject, error) {
- sbParams := params.(*storageman.SStoragePackInstanceBackup)
- packFileName, err := storageman.DoInstancePackBackup(ctx, *sbParams)
- if err != nil {
- return nil, errors.Wrap(err, "DoInstancePackBackup")
- }
- ret := jsonutils.NewDict()
- ret.Set("pack_file_name", jsonutils.NewString(packFileName))
- return ret, nil
- }
- func unpackInstanceBackup(ctx context.Context, params interface{}) (jsonutils.JSONObject, error) {
- sbParams := params.(*storageman.SStorageUnpackInstanceBackup)
- diskBackupIds, metadata, err := storageman.DoInstanceUnpackBackup(ctx, *sbParams)
- if err != nil {
- return nil, errors.Wrap(err, "DoInstanceUnpackBackup")
- }
- ret := jsonutils.NewDict()
- if diskBackupIds != nil {
- ret.Set("disk_backup_ids", jsonutils.Marshal(diskBackupIds))
- }
- ret.Set("metadata", jsonutils.Marshal(metadata))
- return ret, nil
- }
- func storageDeleteBackup(ctx context.Context, w http.ResponseWriter, r *http.Request) {
- _, _, body := appsrv.FetchEnv(ctx, w, r)
- backupId, err := body.GetString("backup_id")
- if err != nil {
- hostutils.Response(ctx, w, httperrors.NewMissingParameterError("backup_id"))
- return
- }
- backupStorageId, err := body.GetString("backup_storage_id")
- if err != nil {
- hostutils.Response(ctx, w, httperrors.NewMissingParameterError("backup_storage_id"))
- return
- }
- backupStorageAccessInfo, err := body.Get("backup_storage_access_info")
- if err != nil {
- hostutils.Response(ctx, w, httperrors.NewMissingParameterError("backup_storage_access_info"))
- return
- }
- hostutils.DelayTask(ctx, deleteBackup, &storageman.SStorageBackup{
- BackupId: backupId,
- BackupStorageId: backupStorageId,
- BackupStorageAccessInfo: backupStorageAccessInfo.(*jsonutils.JSONDict),
- })
- hostutils.ResponseOk(ctx, w)
- }
- func checkOptions(ctx context.Context, w http.ResponseWriter, body jsonutils.JSONObject, options ...string) bool {
- for _, option := range options {
- if body.Contains(option) {
- continue
- }
- hostutils.Response(ctx, w, httperrors.NewMissingParameterError(option))
- return false
- }
- return true
- }
- func deleteBackup(ctx context.Context, params interface{}) (jsonutils.JSONObject, error) {
- sbParams := params.(*storageman.SStorageBackup)
- backupStorage, err := backupstorage.GetBackupStorage(sbParams.BackupStorageId, sbParams.BackupStorageAccessInfo)
- if err != nil {
- return nil, err
- }
- err = backupStorage.RemoveBackup(ctx, sbParams.BackupId)
- if err != nil {
- return nil, err
- }
- return nil, nil
- }
- func storageDeleteSnapshot(ctx context.Context, w http.ResponseWriter, r *http.Request) {
- params, _, body := appsrv.FetchEnv(ctx, w, r)
- var storageId = params["<storageId>"]
- storage := storageman.GetManager().GetStorage(storageId)
- if storage == nil {
- hostutils.Response(ctx, w, httperrors.NewNotFoundError("Stroage Not found"))
- return
- }
- diskId, err := body.GetString("disk_id")
- if err != nil {
- hostutils.Response(ctx, w, httperrors.NewMissingParameterError("disk_id"))
- return
- }
- snapshotId, err := body.GetString("delete_snapshot")
- if err != nil {
- hostutils.Response(ctx, w, httperrors.NewMissingParameterError("snapshot_id"))
- return
- }
- // blockStream indicate snapshot<-disk
- blockStream := jsonutils.QueryBoolean(body, "block_stream", false)
- autoDeleted := jsonutils.QueryBoolean(body, "auto_deleted", false)
- input := &storageman.SStorageDeleteSnapshot{
- DiskId: diskId,
- BlockStream: blockStream,
- SnapshotId: snapshotId,
- }
- if body.Contains("encrypt_info") {
- encryptInfo := apis.SEncryptInfo{}
- if err = body.Unmarshal(&encryptInfo, "encrypt_info"); err != nil {
- hostutils.Response(ctx, w, httperrors.NewInputParameterError("unmarshal encrypt_info failed %s", err))
- return
- }
- input.EncryptInfo = encryptInfo
- }
- if !blockStream && !autoDeleted {
- convertSnapshot, err := body.GetString("convert_snapshot")
- if err != nil {
- hostutils.Response(ctx, w, httperrors.NewMissingParameterError("convert_snapshot"))
- return
- }
- input.ConvertSnapshot = convertSnapshot
- }
- hostutils.DelayTask(ctx, storage.DeleteSnapshot, input)
- hostutils.ResponseOk(ctx, w)
- }
- func storageDeleteSnapshots(ctx context.Context, w http.ResponseWriter, r *http.Request) {
- params, _, body := appsrv.FetchEnv(ctx, w, r)
- var storageId = params["<storageId>"]
- storage := storageman.GetManager().GetStorage(storageId)
- if storage == nil {
- hostutils.Response(ctx, w, httperrors.NewNotFoundError("Storage Not found"))
- return
- }
- diskId, err := body.GetString("disk_id")
- if err != nil {
- hostutils.Response(ctx, w, httperrors.NewImageNotFoundError("disk_id"))
- return
- }
- snapshotIds := []string{}
- err = body.Unmarshal(&snapshotIds, "snapshot_ids")
- if err != nil {
- hostutils.Response(ctx, w, httperrors.NewMissingParameterError("snapshot_ids"))
- return
- }
- input := &storageman.SStorageDeleteSnapshots{
- DiskId: diskId,
- SnapshotIds: snapshotIds,
- }
- hostutils.DelayTask(ctx, storage.DeleteSnapshots, input)
- hostutils.ResponseOk(ctx, w)
- }
- func storageCleanRecycleDiskfiles(ctx context.Context, w http.ResponseWriter, r *http.Request) {
- params, _, _ := appsrv.FetchEnv(ctx, w, r)
- var storageId = params["<storageId>"]
- storage := storageman.GetManager().GetStorage(storageId)
- if storage == nil {
- hostutils.Response(ctx, w, httperrors.NewNotFoundError("Storage Not found"))
- return
- }
- go storage.CleanRecycleDiskfiles(ctx)
- hostutils.ResponseOk(ctx, w)
- }
|