mirror of
https://github.com/yunionio/cloudpods.git
synced 2026-07-02 05:14:20 +08:00
250 lines
7.5 KiB
Go
250 lines
7.5 KiB
Go
// 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 service
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
"time"
|
|
|
|
execlient "yunion.io/x/executor/client"
|
|
"yunion.io/x/jsonutils"
|
|
"yunion.io/x/log"
|
|
"yunion.io/x/pkg/util/qemuimgfmt"
|
|
"yunion.io/x/pkg/utils"
|
|
_ "yunion.io/x/sqlchemy/backends"
|
|
|
|
api "yunion.io/x/onecloud/pkg/apis/image"
|
|
"yunion.io/x/onecloud/pkg/cloudcommon"
|
|
app_common "yunion.io/x/onecloud/pkg/cloudcommon/app"
|
|
"yunion.io/x/onecloud/pkg/cloudcommon/cronman"
|
|
"yunion.io/x/onecloud/pkg/cloudcommon/db"
|
|
common_options "yunion.io/x/onecloud/pkg/cloudcommon/options"
|
|
"yunion.io/x/onecloud/pkg/hostman/hostdeployer/deployclient"
|
|
"yunion.io/x/onecloud/pkg/image/drivers/s3"
|
|
"yunion.io/x/onecloud/pkg/image/models"
|
|
"yunion.io/x/onecloud/pkg/image/options"
|
|
_ "yunion.io/x/onecloud/pkg/image/policy"
|
|
_ "yunion.io/x/onecloud/pkg/image/tasks"
|
|
"yunion.io/x/onecloud/pkg/image/torrent"
|
|
"yunion.io/x/onecloud/pkg/mcclient/auth"
|
|
"yunion.io/x/onecloud/pkg/mcclient/modules/compute"
|
|
"yunion.io/x/onecloud/pkg/util/fileutils2"
|
|
"yunion.io/x/onecloud/pkg/util/procutils"
|
|
)
|
|
|
|
func StartService() {
|
|
opts := &options.Options
|
|
commonOpts := &opts.CommonOptions
|
|
baseOpts := &opts.BaseOptions
|
|
dbOpts := &opts.DBOptions
|
|
common_options.ParseOptions(opts, os.Args, "glance-api.conf", api.SERVICE_TYPE)
|
|
|
|
// no need to run glance as root any more
|
|
// isRoot := sysutils.IsRootPermission()
|
|
// if !isRoot {
|
|
// log.Fatalf("glance service must running with root permissions")
|
|
// }
|
|
|
|
if opts.PortV2 > 0 {
|
|
log.Infof("Port V2 %d is specified, use v2 port", opts.PortV2)
|
|
opts.Port = opts.PortV2
|
|
}
|
|
if len(opts.FilesystemStoreDatadir) == 0 {
|
|
log.Errorf("missing FilesystemStoreDatadir")
|
|
return
|
|
}
|
|
if !fileutils2.Exists(opts.FilesystemStoreDatadir) {
|
|
err := os.MkdirAll(opts.FilesystemStoreDatadir, 0755)
|
|
if err != nil {
|
|
log.Errorf("fail to create %s: %s", opts.FilesystemStoreDatadir, err)
|
|
return
|
|
}
|
|
}
|
|
if len(opts.TorrentStoreDir) == 0 {
|
|
opts.TorrentStoreDir = filepath.Join(filepath.Dir(opts.FilesystemStoreDatadir), "torrents")
|
|
if !fileutils2.Exists(opts.TorrentStoreDir) {
|
|
err := os.MkdirAll(opts.TorrentStoreDir, 0755)
|
|
if err != nil {
|
|
log.Errorf("fail to create %s: %s", opts.TorrentStoreDir, err)
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
log.Infof("exec socket path: %s", options.Options.ExecutorSocketPath)
|
|
if options.Options.EnableRemoteExecutor {
|
|
execlient.Init(options.Options.ExecutorSocketPath)
|
|
execlient.SetTimeoutSeconds(options.Options.ExecutorConnectTimeoutSeconds)
|
|
procutils.SetRemoteExecutor()
|
|
}
|
|
|
|
log.Infof("Target image formats %#v", opts.TargetImageFormats)
|
|
|
|
app_common.InitAuth(commonOpts, func() {
|
|
log.Infof("Auth complete!!")
|
|
})
|
|
|
|
if ok, err := hasVmwareAccount(); err != nil {
|
|
log.Errorf("failed get vmware cloudaccounts")
|
|
} else if ok {
|
|
if !utils.IsInStringArray(string(qemuimgfmt.VMDK), options.Options.TargetImageFormats) {
|
|
options.Options.TargetImageFormats = append(options.Options.TargetImageFormats, string(qemuimgfmt.VMDK))
|
|
}
|
|
}
|
|
|
|
trackers := torrent.GetTrackers()
|
|
if len(trackers) == 0 {
|
|
log.Errorf("no valid torrent-tracker")
|
|
// return
|
|
}
|
|
|
|
app := app_common.InitApp(baseOpts, true)
|
|
|
|
cloudcommon.InitDB(dbOpts)
|
|
|
|
InitHandlers(app)
|
|
|
|
db.EnsureAppSyncDB(app, dbOpts, models.InitDB)
|
|
|
|
common_options.StartOptionManager(opts, opts.ConfigSyncPeriodSeconds, api.SERVICE_TYPE, api.SERVICE_VERSION, options.OnOptionsChange)
|
|
|
|
models.Init(options.Options.StorageDriver)
|
|
|
|
if len(options.Options.DeployServerSocketPath) > 0 {
|
|
log.Infof("deploy server socket path: %s", options.Options.DeployServerSocketPath)
|
|
deployclient.Init(options.Options.DeployServerSocketPath)
|
|
}
|
|
|
|
if options.Options.StorageDriver == api.IMAGE_STORAGE_DRIVER_S3 {
|
|
go initS3()
|
|
} else {
|
|
// Check the images after everything is ready
|
|
go models.CheckImages()
|
|
}
|
|
|
|
if !opts.IsSlaveNode {
|
|
cron := cronman.InitCronJobManager(true, options.Options.CronJobWorkerCount)
|
|
cron.AddJobAtIntervals("CleanPendingDeleteImages", time.Duration(options.Options.PendingDeleteCheckSeconds)*time.Second, models.ImageManager.CleanPendingDeleteImages)
|
|
cron.AddJobAtIntervals("CalculateQuotaUsages", time.Duration(opts.CalculateQuotaUsageIntervalSeconds)*time.Second, models.QuotaManager.CalculateQuotaUsages)
|
|
cron.AddJobAtIntervals("CleanPendingDeleteGuestImages",
|
|
time.Duration(options.Options.PendingDeleteCheckSeconds)*time.Second, models.GuestImageManager.CleanPendingDeleteImages)
|
|
|
|
cron.AddJobEveryFewHour("AutoPurgeSplitable", 4, 30, 0, db.AutoPurgeSplitable, false)
|
|
|
|
cron.Start()
|
|
}
|
|
|
|
app_common.ServeForeverWithCleanup(app, baseOpts, func() {
|
|
cloudcommon.CloseDB()
|
|
|
|
// cron.Stop()
|
|
|
|
if options.Options.EnableTorrentService {
|
|
torrent.StopTorrents()
|
|
}
|
|
if options.Options.StorageDriver == api.IMAGE_STORAGE_DRIVER_S3 {
|
|
procutils.NewCommand("umount", options.Options.S3MountPoint).Run()
|
|
}
|
|
})
|
|
}
|
|
|
|
func hasVmwareAccount() (bool, error) {
|
|
q := jsonutils.NewDict()
|
|
q.Add(jsonutils.NewString("system"), "scope")
|
|
q.Add(jsonutils.NewString("brand"), "VMware")
|
|
res, err := compute.Cloudaccounts.List(auth.GetAdminSession(context.Background(), options.Options.Region), q)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
return res.Total > 0, nil
|
|
}
|
|
|
|
func initS3() {
|
|
url := options.Options.S3Endpoint
|
|
if !strings.HasPrefix(url, "http://") && !strings.HasPrefix(url, "https://") {
|
|
prefix := "http://"
|
|
if options.Options.S3UseSSL {
|
|
prefix = "https://"
|
|
}
|
|
url = prefix + url
|
|
}
|
|
err := s3.Init(
|
|
url,
|
|
options.Options.S3AccessKey,
|
|
options.Options.S3SecretKey,
|
|
options.Options.S3BucketName,
|
|
options.Options.S3UseSSL,
|
|
)
|
|
if err != nil {
|
|
log.Fatalf("failed init s3 client %s", err)
|
|
}
|
|
func() {
|
|
fd, err := os.OpenFile("/tmp/s3-pass", os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0600)
|
|
if err != nil {
|
|
log.Fatalf("failed open s3 pass file %s", err)
|
|
}
|
|
defer fd.Close()
|
|
_, err = fd.WriteString(fmt.Sprintf("%s:%s", options.Options.S3AccessKey, options.Options.S3SecretKey))
|
|
if err != nil {
|
|
log.Fatalf("failed write s3 pass file")
|
|
}
|
|
}()
|
|
cleanS3Dir := func() {
|
|
// check the s3 mount point has been mounted by previous glance instance
|
|
// if it is mounted, just wait
|
|
for {
|
|
if err := procutils.NewRemoteCommandAsFarAsPossible("umount", options.Options.S3MountPoint).Run(); err == nil {
|
|
time.Sleep(time.Second)
|
|
} else {
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
cleanS3Dir()
|
|
if !fileutils2.Exists(options.Options.S3MountPoint) {
|
|
err := os.MkdirAll(options.Options.S3MountPoint, 0755)
|
|
if err != nil {
|
|
log.Fatalf("fail to create %s: %s", options.Options.S3MountPoint, err)
|
|
}
|
|
} else {
|
|
cleanS3Dir()
|
|
}
|
|
|
|
out, err := procutils.NewCommand("s3fs",
|
|
options.Options.S3BucketName, options.Options.S3MountPoint,
|
|
"-o", fmt.Sprintf("passwd_file=/tmp/s3-pass,use_path_request_style,url=%s", url)).Output()
|
|
if err != nil {
|
|
log.Fatalf("failed mount s3fs %s %s", err, out)
|
|
}
|
|
log.Infof("s3fs: %s", out)
|
|
|
|
for {
|
|
if err := procutils.NewRemoteCommandAsFarAsPossible("mountpoint", options.Options.S3MountPoint).Run(); err != nil {
|
|
// sleep 1 second
|
|
time.Sleep(time.Second)
|
|
} else {
|
|
break
|
|
}
|
|
}
|
|
|
|
// check image after s3 mounted
|
|
models.CheckImages()
|
|
}
|