diff --git a/build/scripts/migration/service.d/casaos/migration.list b/build/scripts/migration/service.d/casaos/migration.list index 758088f..24ca8d6 100644 --- a/build/scripts/migration/service.d/casaos/migration.list +++ b/build/scripts/migration/service.d/casaos/migration.list @@ -1,3 +1,4 @@ LEGACY_WITHOUT_VERSION ${DOWNLOAD_DOMAIN}IceWhaleTech/CasaOS/releases/download/v0.3.6/linux-${ARCH}-casaos-migration-tool-v0.3.6.tar.gz v0.3.5 ${DOWNLOAD_DOMAIN}IceWhaleTech/CasaOS/releases/download/v0.3.6/linux-${ARCH}-casaos-migration-tool-v0.3.6.tar.gz -v0.3.5.1 ${DOWNLOAD_DOMAIN}IceWhaleTech/CasaOS/releases/download/v0.3.6/linux-${ARCH}-casaos-migration-tool-v0.3.6.tar.gz \ No newline at end of file +v0.3.5.1 ${DOWNLOAD_DOMAIN}IceWhaleTech/CasaOS/releases/download/v0.3.6/linux-${ARCH}-casaos-migration-tool-v0.3.6.tar.gz +v0.4.2 ${DOWNLOAD_DOMAIN}IceWhaleTech/CasaOS/releases/download/v0.4.3/linux-${ARCH}-casaos-migration-tool-v0.4.3.tar.gz diff --git a/build/sysroot/usr/share/casaos/shell/helper.sh b/build/sysroot/usr/share/casaos/shell/helper.sh index 925ece1..c6f576d 100644 --- a/build/sysroot/usr/share/casaos/shell/helper.sh +++ b/build/sysroot/usr/share/casaos/shell/helper.sh @@ -25,7 +25,7 @@ GetNetCard() { fi else if [ -d "/sys/devices/virtual/net" ] && [ -d "/sys/class/net" ]; then - ls /sys/class/net/ | grep -v "$(ls /sys/devices/virtual/net/)" + ls /sys/class/net/ | grep -v "$(ls /sys/devices/virtual/net/)" -w fi fi } diff --git a/cmd/migration-tool/migration_dummy.go b/cmd/migration-tool/migration_042.go similarity index 73% rename from cmd/migration-tool/migration_dummy.go rename to cmd/migration-tool/migration_042.go index 0492c4b..e8abd44 100644 --- a/cmd/migration-tool/migration_dummy.go +++ b/cmd/migration-tool/migration_042.go @@ -11,8 +11,11 @@ package main import ( + "os" + interfaces "github.com/IceWhaleTech/CasaOS-Common" "github.com/IceWhaleTech/CasaOS-Common/utils/version" + "github.com/IceWhaleTech/CasaOS/pkg/utils/command" ) type migrationTool struct{} @@ -31,15 +34,15 @@ func (u *migrationTool) IsMigrationNeeded() (bool, error) { return false, nil } - if minorVersion > 3 { + if minorVersion > 4 { return false, nil } - if minorVersion == 3 && patchVersion > 5 { + if minorVersion == 4 && patchVersion != 2 { return false, nil } - _logger.Info("Migration is needed for a CasaOS version 0.3.5 and older...") + _logger.Info("Migration is needed for a CasaOS version 0.4.2 ") return true, nil } @@ -48,6 +51,10 @@ func (u *migrationTool) PreMigrate() error { } func (u *migrationTool) Migrate() error { + _logger.Info("Migration is started for a CasaOS version 0.4.2 ") + command.OnlyExec("systemctl stop rclone.service") + os.Remove("/usr/lib/systemd/system/rclone.service") + command.OnlyExec("systemctl daemon-reload") return nil } diff --git a/drivers/all.go b/drivers/all.go deleted file mode 100644 index 3cdfe84..0000000 --- a/drivers/all.go +++ /dev/null @@ -1,12 +0,0 @@ -package drivers - -import ( - _ "github.com/IceWhaleTech/CasaOS/drivers/dropbox" - _ "github.com/IceWhaleTech/CasaOS/drivers/google_drive" -) - -// All do nothing,just for import -// same as _ import -func All() { - -} diff --git a/drivers/base/client.go b/drivers/base/client.go deleted file mode 100644 index 02e314b..0000000 --- a/drivers/base/client.go +++ /dev/null @@ -1,30 +0,0 @@ -package base - -import ( - "net/http" - "time" - - "github.com/go-resty/resty/v2" -) - -var NoRedirectClient *resty.Client -var RestyClient = NewRestyClient() -var HttpClient = &http.Client{} -var UserAgent = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/87.0.4280.88 Safari/537.36" -var DefaultTimeout = time.Second * 30 - -func init() { - NoRedirectClient = resty.New().SetRedirectPolicy( - resty.RedirectPolicyFunc(func(req *http.Request, via []*http.Request) error { - return http.ErrUseLastResponse - }), - ) - NoRedirectClient.SetHeader("user-agent", UserAgent) -} - -func NewRestyClient() *resty.Client { - return resty.New(). - SetHeader("user-agent", UserAgent). - SetRetryCount(3). - SetTimeout(DefaultTimeout) -} diff --git a/drivers/base/types.go b/drivers/base/types.go deleted file mode 100644 index e2757f2..0000000 --- a/drivers/base/types.go +++ /dev/null @@ -1,12 +0,0 @@ -package base - -import "github.com/go-resty/resty/v2" - -type Json map[string]interface{} - -type TokenResp struct { - AccessToken string `json:"access_token"` - RefreshToken string `json:"refresh_token"` -} - -type ReqCallback func(req *resty.Request) diff --git a/drivers/dropbox/drive.go b/drivers/dropbox/drive.go deleted file mode 100644 index c2aff61..0000000 --- a/drivers/dropbox/drive.go +++ /dev/null @@ -1,100 +0,0 @@ -package dropbox - -import ( - "context" - "errors" - "net/http" - - "github.com/IceWhaleTech/CasaOS-Common/utils/logger" - "github.com/IceWhaleTech/CasaOS/internal/driver" - "github.com/IceWhaleTech/CasaOS/model" - "github.com/IceWhaleTech/CasaOS/pkg/utils" - "github.com/go-resty/resty/v2" - "go.uber.org/zap" -) - -type Dropbox struct { - model.Storage - Addition - AccessToken string -} - -func (d *Dropbox) Config() driver.Config { - return config -} - -func (d *Dropbox) GetAddition() driver.Additional { - return &d.Addition -} - -func (d *Dropbox) Init(ctx context.Context) error { - if len(d.RefreshToken) == 0 { - d.getRefreshToken() - } - return d.refreshToken() -} - -func (d *Dropbox) Drop(ctx context.Context) error { - - return nil -} - -func (d *Dropbox) List(ctx context.Context, dir model.Obj, args model.ListArgs) ([]model.Obj, error) { - files, err := d.getFiles(dir.GetID()) - if err != nil { - return nil, err - } - return utils.SliceConvert(files, func(src File) (model.Obj, error) { - return fileToObj(src), nil - }) -} - -func (d *Dropbox) Link(ctx context.Context, file model.Obj, args model.LinkArgs) (*model.Link, error) { - url := "https://content.dropboxapi.com/2/files/download" - link := model.Link{ - URL: url, - Method: http.MethodPost, - Header: http.Header{ - "Authorization": []string{"Bearer " + d.AccessToken}, - "Dropbox-API-Arg": []string{`{"path": "` + file.GetPath() + `"}`}, - }, - } - return &link, nil -} -func (d *Dropbox) GetUserInfo(ctx context.Context) (string, error) { - url := "https://api.dropboxapi.com/2/users/get_current_account" - user := UserInfo{} - resp, err := d.request(url, http.MethodPost, func(req *resty.Request) { - req.SetHeader("Content-Type", "") - }, &user) - if err != nil { - return "", err - } - logger.Info("resp", zap.Any("resp", string(resp))) - return user.Email, nil -} -func (d *Dropbox) MakeDir(ctx context.Context, parentDir model.Obj, dirName string) error { - return nil -} - -func (d *Dropbox) Move(ctx context.Context, srcObj, dstDir model.Obj) error { - return nil -} - -func (d *Dropbox) Rename(ctx context.Context, srcObj model.Obj, newName string) error { - return nil -} - -func (d *Dropbox) Copy(ctx context.Context, srcObj, dstDir model.Obj) error { - return errors.New("not support") -} - -func (d *Dropbox) Remove(ctx context.Context, obj model.Obj) error { - return nil -} - -func (d *Dropbox) Put(ctx context.Context, dstDir model.Obj, stream model.FileStreamer, up driver.UpdateProgress) error { - return nil -} - -var _ driver.Driver = (*Dropbox)(nil) diff --git a/drivers/dropbox/meta.go b/drivers/dropbox/meta.go deleted file mode 100644 index 24d535d..0000000 --- a/drivers/dropbox/meta.go +++ /dev/null @@ -1,33 +0,0 @@ -package dropbox - -import ( - "github.com/IceWhaleTech/CasaOS/internal/driver" - "github.com/IceWhaleTech/CasaOS/internal/op" -) - -const ICONURL = "./img/driver/Dropbox.svg" -const APPKEY = "tciqajyazzdygt9" -const APPSECRET = "e7gtmv441cwdf0n" - -type Addition struct { - driver.RootID - RefreshToken string `json:"refresh_token" required:"true" omit:"true"` - AppKey string `json:"app_key" type:"string" default:"tciqajyazzdygt9" omit:"true"` - AppSecret string `json:"app_secret" type:"string" default:"e7gtmv441cwdf0n" omit:"true"` - OrderDirection string `json:"order_direction" type:"select" options:"asc,desc" omit:"true"` - AuthUrl string `json:"auth_url" type:"string" default:"https://www.dropbox.com/oauth2/authorize?client_id=tciqajyazzdygt9&redirect_uri=https://cloudoauth.files.casaos.app&response_type=code&token_access_type=offline&state=${HOST}%2Fv1%2Frecover%2FDropbox&&force_reapprove=true&force_reauthentication=true"` - Icon string `json:"icon" type:"string" default:"./img/driver/Dropbox.svg"` - Code string `json:"code" type:"string" help:"code from auth_url" omit:"true"` -} - -var config = driver.Config{ - Name: "Dropbox", - OnlyProxy: true, - DefaultRoot: "root", -} - -func init() { - op.RegisterDriver(func() driver.Driver { - return &Dropbox{} - }) -} diff --git a/drivers/dropbox/types.go b/drivers/dropbox/types.go deleted file mode 100644 index af5bdb9..0000000 --- a/drivers/dropbox/types.go +++ /dev/null @@ -1,88 +0,0 @@ -package dropbox - -import ( - "time" - - "github.com/IceWhaleTech/CasaOS-Common/utils/logger" - "github.com/IceWhaleTech/CasaOS/model" - "go.uber.org/zap" -) - -type UserInfo struct { - AccountID string `json:"account_id"` - Name struct { - GivenName string `json:"given_name"` - Surname string `json:"surname"` - FamiliarName string `json:"familiar_name"` - DisplayName string `json:"display_name"` - AbbreviatedName string `json:"abbreviated_name"` - } `json:"name"` - Email string `json:"email"` - EmailVerified bool `json:"email_verified"` - Disabled bool `json:"disabled"` - Country string `json:"country"` - Locale string `json:"locale"` - ReferralLink string `json:"referral_link"` - IsPaired bool `json:"is_paired"` - AccountType struct { - Tag string `json:".tag"` - } `json:"account_type"` - RootInfo struct { - Tag string `json:".tag"` - RootNamespaceID string `json:"root_namespace_id"` - HomeNamespaceID string `json:"home_namespace_id"` - } `json:"root_info"` -} -type TokenError struct { - Error string `json:"error"` - ErrorDescription string `json:"error_description"` -} -type File struct { - Tag string `json:".tag"` - Name string `json:"name"` - PathLower string `json:"path_lower"` - PathDisplay string `json:"path_display"` - ID string `json:"id"` - ClientModified time.Time `json:"client_modified,omitempty"` - ServerModified time.Time `json:"server_modified,omitempty"` - Rev string `json:"rev,omitempty"` - Size int `json:"size,omitempty"` - IsDownloadable bool `json:"is_downloadable,omitempty"` - ContentHash string `json:"content_hash,omitempty"` -} - -type Files struct { - Files []File `json:"entries"` - Cursor string `json:"cursor"` - HasMore bool `json:"has_more"` -} - -type Error struct { - Error struct { - Errors []struct { - Domain string `json:"domain"` - Reason string `json:"reason"` - Message string `json:"message"` - LocationType string `json:"location_type"` - Location string `json:"location"` - } - Code int `json:"code"` - Message string `json:"message"` - } `json:"error"` -} - -func fileToObj(f File) *model.ObjThumb { - logger.Info("dropbox file", zap.Any("file", f)) - obj := &model.ObjThumb{ - Object: model.Object{ - ID: f.ID, - Name: f.Name, - Size: int64(f.Size), - Modified: f.ClientModified, - IsFolder: f.Tag == "folder", - Path: f.PathDisplay, - }, - Thumbnail: model.Thumbnail{}, - } - return obj -} diff --git a/drivers/dropbox/util.go b/drivers/dropbox/util.go deleted file mode 100644 index 7cdf6de..0000000 --- a/drivers/dropbox/util.go +++ /dev/null @@ -1,102 +0,0 @@ -package dropbox - -import ( - "fmt" - "net/http" - - "github.com/IceWhaleTech/CasaOS-Common/utils/logger" - "github.com/IceWhaleTech/CasaOS/drivers/base" - "github.com/go-resty/resty/v2" - "go.uber.org/zap" -) - -func (d *Dropbox) getRefreshToken() error { - url := "https://api.dropbox.com/oauth2/token" - var resp base.TokenResp - var e TokenError - - res, err := base.RestyClient.R().SetResult(&resp).SetError(&e). - SetFormData(map[string]string{ - "code": d.Code, - "grant_type": "authorization_code", - "redirect_uri": "https://cloudoauth.files.casaos.app", - }).SetBasicAuth(d.Addition.AppKey, d.Addition.AppSecret).SetHeader("Content-Type", "application/x-www-form-urlencoded").Post(url) - if err != nil { - return err - } - logger.Info("get refresh token", zap.String("res", res.String())) - if e.Error != "" { - return fmt.Errorf(e.Error) - } - d.RefreshToken = resp.RefreshToken - return nil - -} -func (d *Dropbox) refreshToken() error { - url := "https://api.dropbox.com/oauth2/token" - var resp base.TokenResp - var e TokenError - - res, err := base.RestyClient.R().SetResult(&resp).SetError(&e). - SetFormData(map[string]string{ - "refresh_token": d.RefreshToken, - "grant_type": "refresh_token", - }).SetBasicAuth(d.Addition.AppKey, d.Addition.AppSecret).SetHeader("Content-Type", "application/x-www-form-urlencoded").Post(url) - if err != nil { - return err - } - logger.Info("get refresh token", zap.String("res", res.String())) - if e.Error != "" { - return fmt.Errorf(e.Error) - } - d.AccessToken = resp.AccessToken - return nil - -} -func (d *Dropbox) request(url string, method string, callback base.ReqCallback, resp interface{}) ([]byte, error) { - req := base.RestyClient.R() - req.SetHeader("Authorization", "Bearer "+d.AccessToken) - req.SetHeader("Content-Type", "application/json") - if callback != nil { - callback(req) - } - if resp != nil { - req.SetResult(resp) - } - var e Error - req.SetError(&e) - res, err := req.Execute(method, url) - if err != nil { - return nil, err - } - if e.Error.Code != 0 { - if e.Error.Code == 401 { - err = d.refreshToken() - if err != nil { - return nil, err - } - return d.request(url, method, callback, resp) - } - return nil, fmt.Errorf("%s: %v", e.Error.Message, e.Error.Errors) - } - return res.Body(), nil -} -func (d *Dropbox) getFiles(path string) ([]File, error) { - - res := make([]File, 0) - var resp Files - body := base.Json{ - "limit": 2000, - "path": path, - } - - _, err := d.request("https://api.dropboxapi.com/2/files/list_folder", http.MethodPost, func(req *resty.Request) { - req.SetBody(body) - }, &resp) - if err != nil { - return nil, err - } - res = append(res, resp.Files...) - - return res, nil -} diff --git a/drivers/google_drive/drive.go b/drivers/google_drive/drive.go deleted file mode 100644 index 9bad0c2..0000000 --- a/drivers/google_drive/drive.go +++ /dev/null @@ -1,183 +0,0 @@ -package google_drive - -import ( - "context" - "errors" - "fmt" - "net/http" - "strconv" - - "github.com/IceWhaleTech/CasaOS-Common/utils/logger" - "github.com/IceWhaleTech/CasaOS/drivers/base" - "github.com/IceWhaleTech/CasaOS/internal/driver" - "github.com/IceWhaleTech/CasaOS/model" - "github.com/IceWhaleTech/CasaOS/pkg/utils" - "github.com/go-resty/resty/v2" - "go.uber.org/zap" -) - -type GoogleDrive struct { - model.Storage - Addition - AccessToken string -} - -func (d *GoogleDrive) Config() driver.Config { - return config -} - -func (d *GoogleDrive) GetAddition() driver.Additional { - return &d.Addition -} - -func (d *GoogleDrive) Init(ctx context.Context) error { - if d.ChunkSize == 0 { - d.ChunkSize = 5 - } - if len(d.RefreshToken) == 0 { - d.getRefreshToken() - } - return d.refreshToken() -} - -func (d *GoogleDrive) Drop(ctx context.Context) error { - return nil -} - -func (d *GoogleDrive) List(ctx context.Context, dir model.Obj, args model.ListArgs) ([]model.Obj, error) { - files, err := d.getFiles(dir.GetID()) - if err != nil { - return nil, err - } - return utils.SliceConvert(files, func(src File) (model.Obj, error) { - return fileToObj(src), nil - }) -} - -func (d *GoogleDrive) Link(ctx context.Context, file model.Obj, args model.LinkArgs) (*model.Link, error) { - url := fmt.Sprintf("https://www.googleapis.com/drive/v3/files/%s?includeItemsFromAllDrives=true&supportsAllDrives=true", file.GetID()) - _, err := d.request(url, http.MethodGet, nil, nil) - if err != nil { - return nil, err - } - link := model.Link{ - Method: http.MethodGet, - URL: url + "&alt=media", - Header: http.Header{ - "Authorization": []string{"Bearer " + d.AccessToken}, - }, - } - return &link, nil -} -func (d *GoogleDrive) GetUserInfo(ctx context.Context) (string, error) { - url := "https://content.googleapis.com/drive/v3/about?fields=user" - user := UserInfo{} - resp, err := d.request(url, http.MethodGet, nil, &user) - if err != nil { - return "", err - } - logger.Info("resp", zap.Any("resp", resp)) - return user.User.EmailAddress, nil -} - -func (d *GoogleDrive) MakeDir(ctx context.Context, parentDir model.Obj, dirName string) error { - data := base.Json{ - "name": dirName, - "parents": []string{parentDir.GetID()}, - "mimeType": "application/vnd.google-apps.folder", - } - _, err := d.request("https://www.googleapis.com/drive/v3/files", http.MethodPost, func(req *resty.Request) { - req.SetBody(data) - }, nil) - return err -} - -func (d *GoogleDrive) Move(ctx context.Context, srcObj, dstDir model.Obj) error { - query := map[string]string{ - "addParents": dstDir.GetID(), - "removeParents": "root", - } - url := "https://www.googleapis.com/drive/v3/files/" + srcObj.GetID() - _, err := d.request(url, http.MethodPatch, func(req *resty.Request) { - req.SetQueryParams(query) - }, nil) - return err -} - -func (d *GoogleDrive) Rename(ctx context.Context, srcObj model.Obj, newName string) error { - data := base.Json{ - "name": newName, - } - url := "https://www.googleapis.com/drive/v3/files/" + srcObj.GetID() - _, err := d.request(url, http.MethodPatch, func(req *resty.Request) { - req.SetBody(data) - }, nil) - return err -} - -func (d *GoogleDrive) Copy(ctx context.Context, srcObj, dstDir model.Obj) error { - return errors.New("not support") -} - -func (d *GoogleDrive) Remove(ctx context.Context, obj model.Obj) error { - url := "https://www.googleapis.com/drive/v3/files/" + obj.GetID() - _, err := d.request(url, http.MethodDelete, nil, nil) - return err -} - -func (d *GoogleDrive) Put(ctx context.Context, dstDir model.Obj, stream model.FileStreamer, up driver.UpdateProgress) error { - obj := stream.GetOld() - var ( - e Error - url string - data base.Json - res *resty.Response - err error - ) - if obj != nil { - url = fmt.Sprintf("https://www.googleapis.com/upload/drive/v3/files/%s?uploadType=resumable&supportsAllDrives=true", obj.GetID()) - data = base.Json{} - } else { - data = base.Json{ - "name": stream.GetName(), - "parents": []string{dstDir.GetID()}, - } - url = "https://www.googleapis.com/upload/drive/v3/files?uploadType=resumable&supportsAllDrives=true" - } - req := base.NoRedirectClient.R(). - SetHeaders(map[string]string{ - "Authorization": "Bearer " + d.AccessToken, - "X-Upload-Content-Type": stream.GetMimetype(), - "X-Upload-Content-Length": strconv.FormatInt(stream.GetSize(), 10), - }). - SetError(&e).SetBody(data).SetContext(ctx) - if obj != nil { - res, err = req.Patch(url) - } else { - res, err = req.Post(url) - } - if err != nil { - return err - } - if e.Error.Code != 0 { - if e.Error.Code == 401 { - err = d.refreshToken() - if err != nil { - return err - } - return d.Put(ctx, dstDir, stream, up) - } - return fmt.Errorf("%s: %v", e.Error.Message, e.Error.Errors) - } - putUrl := res.Header().Get("location") - if stream.GetSize() < d.ChunkSize*1024*1024 { - _, err = d.request(putUrl, http.MethodPut, func(req *resty.Request) { - req.SetHeader("Content-Length", strconv.FormatInt(stream.GetSize(), 10)).SetBody(stream.GetReadCloser()) - }, nil) - } else { - err = d.chunkUpload(ctx, stream, putUrl) - } - return err -} - -var _ driver.Driver = (*GoogleDrive)(nil) diff --git a/drivers/google_drive/meta.go b/drivers/google_drive/meta.go deleted file mode 100644 index f771966..0000000 --- a/drivers/google_drive/meta.go +++ /dev/null @@ -1,35 +0,0 @@ -package google_drive - -import ( - "github.com/IceWhaleTech/CasaOS/internal/driver" - "github.com/IceWhaleTech/CasaOS/internal/op" -) - -const ICONURL = "./img/driver/GoogleDrive.svg" -const CLIENTID = "921743327851-urr4f7jjfp4ts639evqb3i4m4qb4u4cc.apps.googleusercontent.com" -const CLIENTSECRET = "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" - -type Addition struct { - driver.RootID - RefreshToken string `json:"refresh_token" required:"true" omit:"true"` - OrderBy string `json:"order_by" type:"string" help:"such as: folder,name,modifiedTime" omit:"true"` - OrderDirection string `json:"order_direction" type:"select" options:"asc,desc" omit:"true"` - ClientID string `json:"client_id" required:"true" default:"921743327851-urr4f7jjfp4ts639evqb3i4m4qb4u4cc.apps.googleusercontent.com" omit:"true"` - ClientSecret string `json:"client_secret" required:"true" default:"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" omit:"true"` - ChunkSize int64 `json:"chunk_size" type:"number" help:"chunk size while uploading (unit: MB)" omit:"true"` - AuthUrl string `json:"auth_url" type:"string" default:"https://accounts.google.com/o/oauth2/auth/oauthchooseaccount?response_type=code&client_id=921743327851-urr4f7jjfp4ts639evqb3i4m4qb4u4cc.apps.googleusercontent.com&redirect_uri=https%3A%2F%2Fcloudoauth.files.casaos.app&scope=https%3A%2F%2Fwww.googleapis.com%2Fauth%2Fdrive&access_type=offline&approval_prompt=force&state=${HOST}%2Fv1%2Frecover%2FGoogleDrive&service=lso&o2v=1&flowName=GeneralOAuthFlow"` - Icon string `json:"icon" type:"string" default:"./img/driver/GoogleDrive.svg"` - Code string `json:"code" type:"string" help:"code from auth_url" omit:"true"` -} - -var config = driver.Config{ - Name: "GoogleDrive", - OnlyProxy: true, - DefaultRoot: "root", -} - -func init() { - op.RegisterDriver(func() driver.Driver { - return &GoogleDrive{} - }) -} diff --git a/drivers/google_drive/types.go b/drivers/google_drive/types.go deleted file mode 100644 index 4bbab52..0000000 --- a/drivers/google_drive/types.go +++ /dev/null @@ -1,77 +0,0 @@ -package google_drive - -import ( - "strconv" - "time" - - "github.com/IceWhaleTech/CasaOS/model" - log "github.com/sirupsen/logrus" -) - -type UserInfo struct { - User struct { - Kind string `json:"kind"` - DisplayName string `json:"displayName"` - PhotoLink string `json:"photoLink"` - Me bool `json:"me"` - PermissionID string `json:"permissionId"` - EmailAddress string `json:"emailAddress"` - } `json:"user"` -} - -type TokenError struct { - Error string `json:"error"` - ErrorDescription string `json:"error_description"` -} - -type Files struct { - NextPageToken string `json:"nextPageToken"` - Files []File `json:"files"` -} - -type File struct { - Id string `json:"id"` - Name string `json:"name"` - MimeType string `json:"mimeType"` - ModifiedTime time.Time `json:"modifiedTime"` - Size string `json:"size"` - ThumbnailLink string `json:"thumbnailLink"` - ShortcutDetails struct { - TargetId string `json:"targetId"` - TargetMimeType string `json:"targetMimeType"` - } `json:"shortcutDetails"` -} - -func fileToObj(f File) *model.ObjThumb { - log.Debugf("google file: %+v", f) - size, _ := strconv.ParseInt(f.Size, 10, 64) - obj := &model.ObjThumb{ - Object: model.Object{ - ID: f.Id, - Name: f.Name, - Size: size, - Modified: f.ModifiedTime, - IsFolder: f.MimeType == "application/vnd.google-apps.folder", - }, - Thumbnail: model.Thumbnail{}, - } - if f.MimeType == "application/vnd.google-apps.shortcut" { - obj.ID = f.ShortcutDetails.TargetId - obj.IsFolder = f.ShortcutDetails.TargetMimeType == "application/vnd.google-apps.folder" - } - return obj -} - -type Error struct { - Error struct { - Errors []struct { - Domain string `json:"domain"` - Reason string `json:"reason"` - Message string `json:"message"` - LocationType string `json:"location_type"` - Location string `json:"location"` - } - Code int `json:"code"` - Message string `json:"message"` - } `json:"error"` -} diff --git a/drivers/google_drive/util.go b/drivers/google_drive/util.go deleted file mode 100644 index a3b443a..0000000 --- a/drivers/google_drive/util.go +++ /dev/null @@ -1,152 +0,0 @@ -package google_drive - -import ( - "context" - "fmt" - "io" - "net/http" - "strconv" - - "github.com/IceWhaleTech/CasaOS-Common/utils/logger" - "github.com/IceWhaleTech/CasaOS/drivers/base" - "github.com/IceWhaleTech/CasaOS/model" - "github.com/IceWhaleTech/CasaOS/pkg/utils" - "github.com/go-resty/resty/v2" - log "github.com/sirupsen/logrus" - "go.uber.org/zap" -) - -// do others that not defined in Driver interface - -func (d *GoogleDrive) getRefreshToken() error { - url := "https://www.googleapis.com/oauth2/v4/token" - var resp base.TokenResp - var e TokenError - res, err := base.RestyClient.R().SetResult(&resp).SetError(&e). - SetFormData(map[string]string{ - "client_id": d.ClientID, - "client_secret": d.ClientSecret, - "code": d.Code, - "grant_type": "authorization_code", - "redirect_uri": "https://cloudoauth.files.casaos.app", - }).Post(url) - if err != nil { - return err - } - logger.Info("get refresh token", zap.String("res", res.String())) - if e.Error != "" { - return fmt.Errorf(e.Error) - } - d.RefreshToken = resp.RefreshToken - return nil -} - -func (d *GoogleDrive) refreshToken() error { - url := "https://www.googleapis.com/oauth2/v4/token" - var resp base.TokenResp - var e TokenError - res, err := base.RestyClient.R().SetResult(&resp).SetError(&e). - SetFormData(map[string]string{ - "client_id": d.ClientID, - "client_secret": d.ClientSecret, - "refresh_token": d.RefreshToken, - "grant_type": "refresh_token", - }).Post(url) - if err != nil { - return err - } - log.Debug(res.String()) - if e.Error != "" { - return fmt.Errorf(e.Error) - } - d.AccessToken = resp.AccessToken - return nil -} - -func (d *GoogleDrive) request(url string, method string, callback base.ReqCallback, resp interface{}) ([]byte, error) { - req := base.RestyClient.R() - req.SetHeader("Authorization", "Bearer "+d.AccessToken) - req.SetQueryParam("includeItemsFromAllDrives", "true") - req.SetQueryParam("supportsAllDrives", "true") - if callback != nil { - callback(req) - } - if resp != nil { - req.SetResult(resp) - } - var e Error - req.SetError(&e) - res, err := req.Execute(method, url) - if err != nil { - return nil, err - } - if e.Error.Code != 0 { - if e.Error.Code == 401 { - err = d.refreshToken() - if err != nil { - return nil, err - } - return d.request(url, method, callback, resp) - } - return nil, fmt.Errorf("%s: %v", e.Error.Message, e.Error.Errors) - } - return res.Body(), nil -} - -func (d *GoogleDrive) getFiles(id string) ([]File, error) { - pageToken := "first" - res := make([]File, 0) - for pageToken != "" { - if pageToken == "first" { - pageToken = "" - } - var resp Files - orderBy := "folder,name,modifiedTime desc" - if d.OrderBy != "" { - orderBy = d.OrderBy + " " + d.OrderDirection - } - query := map[string]string{ - "orderBy": orderBy, - "fields": "files(id,name,mimeType,size,modifiedTime,thumbnailLink,shortcutDetails),nextPageToken", - "pageSize": "1000", - "q": fmt.Sprintf("'%s' in parents and trashed = false", id), - //"includeItemsFromAllDrives": "true", - //"supportsAllDrives": "true", - "pageToken": pageToken, - } - _, err := d.request("https://www.googleapis.com/drive/v3/files", http.MethodGet, func(req *resty.Request) { - req.SetQueryParams(query) - }, &resp) - if err != nil { - return nil, err - } - pageToken = resp.NextPageToken - res = append(res, resp.Files...) - } - return res, nil -} - -func (d *GoogleDrive) chunkUpload(ctx context.Context, stream model.FileStreamer, url string) error { - var defaultChunkSize = d.ChunkSize * 1024 * 1024 - var finish int64 = 0 - for finish < stream.GetSize() { - if utils.IsCanceled(ctx) { - return ctx.Err() - } - chunkSize := stream.GetSize() - finish - if chunkSize > defaultChunkSize { - chunkSize = defaultChunkSize - } - _, err := d.request(url, http.MethodPut, func(req *resty.Request) { - req.SetHeaders(map[string]string{ - "Content-Length": strconv.FormatInt(chunkSize, 10), - "Content-Range": fmt.Sprintf("bytes %d-%d/%d", finish, finish+chunkSize-1, stream.GetSize()), - }).SetBody(io.LimitReader(stream.GetReadCloser(), chunkSize)).SetContext(ctx) - }, nil) - if err != nil { - return err - } - finish += chunkSize - } - return nil -} diff --git a/go.mod b/go.mod index eb258f6..305d7a6 100644 --- a/go.mod +++ b/go.mod @@ -3,6 +3,7 @@ module github.com/IceWhaleTech/CasaOS go 1.19 require ( + bazil.org/fuse v0.0.0-20200524192727-fb710f7dfd05 github.com/Curtis-Milo/nat-type-identifier-go v0.0.0-20220215191915-18d42168c63d github.com/IceWhaleTech/CasaOS-Common v0.4.2-alpha3 github.com/Xhofe/go-cache v0.0.0-20220723083548-714439c8af9a @@ -33,6 +34,7 @@ require ( github.com/moby/sys/mountinfo v0.6.2 github.com/patrickmn/go-cache v2.1.0+incompatible github.com/pkg/errors v0.9.1 + github.com/rclone/rclone v1.61.1 github.com/robfig/cron v1.2.0 github.com/satori/go.uuid v1.2.0 github.com/shirou/gopsutil/v3 v3.22.11 @@ -42,22 +44,45 @@ require ( go.uber.org/zap v1.24.0 golang.org/x/crypto v0.5.0 golang.org/x/oauth2 v0.3.0 - golang.org/x/sync v0.0.0-20210220032951-036812b2e83c + golang.org/x/sync v0.1.0 gorm.io/gorm v1.24.3 gotest.tools v2.2.0+incompatible ) require ( + cloud.google.com/go/compute v1.12.1 // indirect + cloud.google.com/go/compute/metadata v0.2.1 // indirect + github.com/Azure/azure-sdk-for-go/sdk/azcore v1.1.4 // indirect + github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.2.0 // indirect + github.com/Azure/azure-sdk-for-go/sdk/internal v1.0.1 // indirect + github.com/Azure/azure-sdk-for-go/sdk/storage/azblob v0.5.1 // indirect + github.com/Azure/go-ntlmssp v0.0.0-20220621081337-cb9428e4ac1e // indirect + github.com/AzureAD/microsoft-authentication-library-for-go v0.7.0 // indirect + github.com/Max-Sum/base32768 v0.0.0-20191205131208-7937843c71d5 // indirect + github.com/Microsoft/go-winio v0.5.2 // indirect + github.com/Unknwon/goconfig v1.0.0 // indirect + github.com/abbot/go-http-auth v0.4.0 // indirect github.com/andybalholm/brotli v1.0.1 // indirect github.com/apapsch/go-jsonmerge/v2 v2.0.0 // indirect + github.com/artyom/mtab v1.0.0 // indirect + github.com/aws/aws-sdk-go v1.44.145 // indirect github.com/benbjohnson/clock v1.3.0 // indirect + github.com/beorn7/perks v1.0.1 // indirect + github.com/buengese/sgzip v0.1.1 // indirect + github.com/calebcase/tmpfile v1.0.3 // indirect + github.com/cespare/xxhash/v2 v2.1.2 // indirect + github.com/colinmarc/hdfs/v2 v2.3.0 // indirect + github.com/coreos/go-semver v0.3.0 // indirect github.com/coreos/go-systemd/v22 v22.3.2 // indirect github.com/davecgh/go-spew v1.1.1 // indirect + github.com/dropbox/dropbox-sdk-go-unofficial/v6 v6.0.5 // indirect github.com/dsnet/compress v0.0.2-0.20210315054119-f66993602bf5 // indirect github.com/dsoprea/go-utility/v2 v2.0.0-20221003172846-a3e1774ef349 // indirect + github.com/gabriel-vasile/mimetype v1.4.1 // indirect github.com/geoffgarside/ber v1.1.0 // indirect github.com/gin-contrib/sse v0.1.0 // indirect github.com/glebarez/go-sqlite v1.20.0 // indirect + github.com/go-chi/chi/v5 v5.0.7 // indirect github.com/go-errors/errors v1.4.2 // indirect github.com/go-ole/go-ole v1.2.6 // indirect github.com/go-openapi/jsonpointer v0.19.5 // indirect @@ -67,22 +92,44 @@ require ( github.com/go-playground/validator/v10 v10.11.1 // indirect github.com/goccy/go-json v0.9.11 // indirect github.com/godbus/dbus/v5 v5.0.4 // indirect + github.com/gofrs/flock v0.8.1 // indirect github.com/gofrs/uuid v4.0.0+incompatible // indirect + github.com/gogo/protobuf v1.3.2 // indirect github.com/golang-jwt/jwt v3.2.2+incompatible // indirect github.com/golang-jwt/jwt/v4 v4.4.2 // indirect github.com/golang/geo v0.0.0-20210211234256-740aa86cb551 // indirect + github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da // indirect github.com/golang/protobuf v1.5.2 // indirect github.com/golang/snappy v0.0.4 // indirect github.com/google/go-cmp v0.5.9 // indirect - github.com/google/go-querystring v1.0.0 // indirect + github.com/google/go-querystring v1.1.0 // indirect github.com/google/uuid v1.3.0 // indirect + github.com/googleapis/enterprise-certificate-proxy v0.2.0 // indirect + github.com/googleapis/gax-go/v2 v2.7.0 // indirect github.com/gorilla/mux v1.8.0 // indirect + github.com/hashicorp/errwrap v1.0.0 // indirect + github.com/hashicorp/go-multierror v1.1.1 // indirect + github.com/hashicorp/go-uuid v1.0.3 // indirect + github.com/iguanesolutions/go-systemd/v5 v5.1.0 // indirect + github.com/inconshreveable/mousetrap v1.0.1 // indirect github.com/invopop/yaml v0.1.0 // indirect + github.com/jcmturner/aescts/v2 v2.0.0 // indirect + github.com/jcmturner/dnsutils/v2 v2.0.0 // indirect + github.com/jcmturner/gofork v1.7.6 // indirect + github.com/jcmturner/goidentity/v6 v6.0.1 // indirect + github.com/jcmturner/gokrb5/v8 v8.4.3 // indirect + github.com/jcmturner/rpc/v2 v2.0.3 // indirect github.com/jinzhu/inflection v1.0.0 // indirect github.com/jinzhu/now v1.1.5 // indirect + github.com/jmespath/go-jmespath v0.4.0 // indirect github.com/josharian/intern v1.0.0 // indirect + github.com/jzelinskie/whirlpool v0.0.0-20201016144138-0675e54bb004 // indirect github.com/klauspost/compress v1.15.13 // indirect github.com/klauspost/pgzip v1.2.5 // indirect + github.com/koofr/go-httpclient v0.0.0-20200420163713-93aa7c75b348 // indirect + github.com/koofr/go-koofrclient v0.0.0-20190724113126-8e5366da203a // indirect + github.com/kr/fs v0.1.0 // indirect + github.com/kylelemons/godebug v1.1.0 // indirect github.com/labstack/gommon v0.4.0 // indirect github.com/leodido/go-urn v1.2.1 // indirect github.com/lufia/plan9stats v0.0.0-20211012122336-39d0f177ccd0 // indirect @@ -90,16 +137,39 @@ require ( github.com/mattn/go-colorable v0.1.13 // indirect github.com/mattn/go-isatty v0.0.16 // indirect github.com/mattn/go-sqlite3 v1.14.15 // indirect + github.com/matttproud/golang_protobuf_extensions v1.0.1 // indirect + github.com/mitchellh/go-homedir v1.1.0 // indirect github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect github.com/modern-go/reflect2 v1.0.2 // indirect github.com/mohae/deepcopy v0.0.0-20170929034955-c48cc78d4826 // indirect + github.com/ncw/go-acd v0.0.0-20201019170801-fe55f33415b1 // indirect + github.com/ncw/swift/v2 v2.0.1 // indirect github.com/nwaples/rardecode v1.1.0 // indirect + github.com/oracle/oci-go-sdk/v65 v65.26.1 // indirect github.com/pelletier/go-toml/v2 v2.0.6 // indirect + github.com/pengsrc/go-shared v0.2.1-0.20190131101655-1999055a4a14 // indirect github.com/perimeterx/marshmallow v1.1.4 // indirect github.com/pierrec/lz4/v4 v4.1.2 // indirect + github.com/pkg/browser v0.0.0-20210115035449-ce105d075bb4 // indirect + github.com/pkg/sftp v1.13.5 // indirect + github.com/pkg/xattr v0.4.9 // indirect github.com/pmezard/go-difflib v1.0.0 // indirect github.com/power-devops/perfstat v0.0.0-20210106213030-5aafc221ea8c // indirect + github.com/prometheus/client_golang v1.14.0 // indirect + github.com/prometheus/client_model v0.3.0 // indirect + github.com/prometheus/common v0.37.0 // indirect + github.com/prometheus/procfs v0.8.0 // indirect + github.com/putdotio/go-putio/putio v0.0.0-20200123120452-16d982cac2b8 // indirect + github.com/rclone/ftp v0.0.0-20221014110213-e44dedbc76c6 // indirect github.com/remyoudompheng/bigfft v0.0.0-20200410134404-eec4a21b6bb0 // indirect + github.com/rfjakob/eme v1.1.2 // indirect + github.com/skratchdot/open-golang v0.0.0-20200116055534-eef842397966 // indirect + github.com/smartystreets/goconvey v1.7.2 // indirect + github.com/sony/gobreaker v0.5.0 // indirect + github.com/spacemonkeygo/monkit/v3 v3.0.17 // indirect + github.com/spf13/cobra v1.6.1 // indirect + github.com/spf13/pflag v1.0.5 // indirect + github.com/t3rm1n4l/go-mega v0.0.0-20220725095014-c4e0c2b5debf // indirect github.com/tidwall/match v1.1.1 // indirect github.com/tidwall/pretty v1.2.0 // indirect github.com/tklauser/go-sysconf v0.3.11 // indirect @@ -108,16 +178,27 @@ require ( github.com/ulikunitz/xz v0.5.9 // indirect github.com/valyala/bytebufferpool v1.0.0 // indirect github.com/valyala/fasttemplate v1.2.2 // indirect + github.com/vivint/infectious v0.0.0-20200605153912-25a574ae18a3 // indirect + github.com/xanzy/ssh-agent v0.3.3 // indirect github.com/xi2/xz v0.0.0-20171230120015-48954b6210f8 // indirect + github.com/youmark/pkcs8 v0.0.0-20201027041543-1326539a0a0a // indirect + github.com/yunify/qingstor-sdk-go/v3 v3.2.0 // indirect github.com/yusufpapurcu/wmi v1.2.2 // indirect + github.com/zeebo/errs v1.3.0 // indirect + go.etcd.io/bbolt v1.3.6 // indirect + go.opencensus.io v0.24.0 // indirect go.uber.org/atomic v1.7.0 // indirect go.uber.org/multierr v1.6.0 // indirect golang.org/x/image v0.0.0-20191009234506-e7c1f5e7dbb8 // indirect golang.org/x/net v0.5.0 // indirect golang.org/x/sys v0.4.0 // indirect + golang.org/x/term v0.4.0 // indirect golang.org/x/text v0.6.0 // indirect golang.org/x/time v0.2.0 // indirect + google.golang.org/api v0.103.0 // indirect google.golang.org/appengine v1.6.7 // indirect + google.golang.org/genproto v0.0.0-20221027153422-115e99e71e1c // indirect + google.golang.org/grpc v1.50.1 // indirect google.golang.org/protobuf v1.28.1 // indirect gopkg.in/ini.v1 v1.67.0 // indirect gopkg.in/natefinch/lumberjack.v2 v2.0.0 // indirect @@ -127,4 +208,7 @@ require ( modernc.org/mathutil v1.5.0 // indirect modernc.org/memory v1.4.0 // indirect modernc.org/sqlite v1.20.0 // indirect + storj.io/common v0.0.0-20220414110316-a5cb7172d6bf // indirect + storj.io/drpc v0.0.30 // indirect + storj.io/uplink v1.9.0 // indirect ) diff --git a/go.sum b/go.sum index 46a5eba..e255efa 100644 --- a/go.sum +++ b/go.sum @@ -1,25 +1,121 @@ +bazil.org/fuse v0.0.0-20200524192727-fb710f7dfd05 h1:UrYe9YkT4Wpm6D+zByEyCJQzDqTPXqTDUI7bZ41i9VE= +bazil.org/fuse v0.0.0-20200524192727-fb710f7dfd05/go.mod h1:h0h5FBYpXThbvSfTqthw+0I4nmHnhTHkO5BoOHsBWqg= +cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= +cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= +cloud.google.com/go v0.38.0/go.mod h1:990N+gfupTy94rShfmMCWGDn0LpTmnzTp2qbd1dvSRU= +cloud.google.com/go v0.44.1/go.mod h1:iSa0KzasP4Uvy3f1mN/7PiObzGgflwredwwASm/v6AU= +cloud.google.com/go v0.44.2/go.mod h1:60680Gw3Yr4ikxnPRS/oxxkBccT6SA1yMk63TGekxKY= +cloud.google.com/go v0.45.1/go.mod h1:RpBamKRgapWJb87xiFSdk4g1CME7QZg3uwTez+TSTjc= +cloud.google.com/go v0.46.3/go.mod h1:a6bKKbmY7er1mI7TEI4lsAkts/mkhTSZK8w33B4RAg0= +cloud.google.com/go v0.50.0/go.mod h1:r9sluTvynVuxRIOHXQEHMFffphuXHOMZMycpNR5e6To= +cloud.google.com/go v0.52.0/go.mod h1:pXajvRH/6o3+F9jDHZWQ5PbGhn+o8w9qiu/CffaVdO4= +cloud.google.com/go v0.53.0/go.mod h1:fp/UouUEsRkN6ryDKNW/Upv/JBKnv6WDthjR6+vze6M= +cloud.google.com/go v0.54.0/go.mod h1:1rq2OEkV3YMf6n/9ZvGWI3GWw0VoqH/1x2nd8Is/bPc= +cloud.google.com/go v0.56.0/go.mod h1:jr7tqZxxKOVYizybht9+26Z/gUq7tiRzu+ACVAMbKVk= +cloud.google.com/go v0.57.0/go.mod h1:oXiQ6Rzq3RAkkY7N6t3TcE6jE+CIBBbA36lwQ1JyzZs= +cloud.google.com/go v0.62.0/go.mod h1:jmCYTdRCQuc1PHIIJ/maLInMho30T/Y0M4hTdTShOYc= +cloud.google.com/go v0.65.0/go.mod h1:O5N8zS7uWy9vkA9vayVHs65eM1ubvY4h553ofrNHObY= +cloud.google.com/go v0.105.0 h1:DNtEKRBAAzeS4KyIory52wWHuClNaXJ5x1F7xa4q+5Y= +cloud.google.com/go/bigquery v1.0.1/go.mod h1:i/xbL2UlR5RvWAURpBYZTtm/cXjCha9lbfbpx4poX+o= +cloud.google.com/go/bigquery v1.3.0/go.mod h1:PjpwJnslEMmckchkHFfq+HTD2DmtT67aNFKH1/VBDHE= +cloud.google.com/go/bigquery v1.4.0/go.mod h1:S8dzgnTigyfTmLBfrtrhyYhwRxG72rYxvftPBK2Dvzc= +cloud.google.com/go/bigquery v1.5.0/go.mod h1:snEHRnqQbz117VIFhE8bmtwIDY80NLUZUMb4Nv6dBIg= +cloud.google.com/go/bigquery v1.7.0/go.mod h1://okPTzCYNXSlb24MZs83e2Do+h+VXtc4gLoIoXIAPc= +cloud.google.com/go/bigquery v1.8.0/go.mod h1:J5hqkt3O0uAFnINi6JXValWIb1v0goeZM77hZzJN/fQ= +cloud.google.com/go/compute v1.12.1 h1:gKVJMEyqV5c/UnpzjjQbo3Rjvvqpr9B1DFSbJC4OXr0= +cloud.google.com/go/compute v1.12.1/go.mod h1:e8yNOBcBONZU1vJKCvCoDw/4JQsA0dpM4x/6PIIOocU= +cloud.google.com/go/compute/metadata v0.2.1 h1:efOwf5ymceDhK6PKMnnrTHP4pppY5L22mle96M1yP48= +cloud.google.com/go/compute/metadata v0.2.1/go.mod h1:jgHgmJd2RKBGzXqF5LR2EZMGxBkeanZ9wwa75XHJgOM= +cloud.google.com/go/datastore v1.0.0/go.mod h1:LXYbyblFSglQ5pkeyhO+Qmw7ukd3C+pD7TKLgZqpHYE= +cloud.google.com/go/datastore v1.1.0/go.mod h1:umbIZjpQpHh4hmRpGhH4tLFup+FVzqBi1b3c64qFpCk= +cloud.google.com/go/longrunning v0.1.1 h1:y50CXG4j0+qvEukslYFBCrzaXX0qpFbBzc3PchSu/LE= +cloud.google.com/go/pubsub v1.0.1/go.mod h1:R0Gpsv3s54REJCy4fxDixWD93lHJMoZTyQ2kNxGRt3I= +cloud.google.com/go/pubsub v1.1.0/go.mod h1:EwwdRX2sKPjnvnqCa270oGRyludottCI76h+R3AArQw= +cloud.google.com/go/pubsub v1.2.0/go.mod h1:jhfEVHT8odbXTkndysNHCcx0awwzvfOlguIAii9o8iA= +cloud.google.com/go/pubsub v1.3.1/go.mod h1:i+ucay31+CNRpDW4Lu78I4xXG+O1r/MAHgjpRVR+TSU= +cloud.google.com/go/storage v1.0.0/go.mod h1:IhtSnM/ZTZV8YYJWCY8RULGVqBDmpoyjwiyrjsg+URw= +cloud.google.com/go/storage v1.5.0/go.mod h1:tpKbwo567HUNpVclU5sGELwQWBDZ8gh0ZeosJ0Rtdos= +cloud.google.com/go/storage v1.6.0/go.mod h1:N7U0C8pVQ/+NIKOBQyamJIeKQKkZ+mxpohlUTyfDhBk= +cloud.google.com/go/storage v1.8.0/go.mod h1:Wv1Oy7z6Yz3DshWRJFhqM/UCfaWIRTdp0RXyy7KQOVs= +cloud.google.com/go/storage v1.10.0/go.mod h1:FLPqc6j+Ki4BU591ie1oL6qBQGu2Bl/tZ9ullr3+Kg0= +dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU= +github.com/Azure/azure-sdk-for-go/sdk/azcore v1.1.4 h1:pqrAR74b6EoR4kcxF7L7Wg2B8Jgil9UUZtMvxhEFqWo= +github.com/Azure/azure-sdk-for-go/sdk/azcore v1.1.4/go.mod h1:uGG2W01BaETf0Ozp+QxxKJdMBNRWPdstHG0Fmdwn1/U= +github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.2.0 h1:t/W5MYAuQy81cvM8VUNfRLzhtKpXhVUAN7Cd7KVbTyc= +github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.2.0/go.mod h1:NBanQUfSWiWn3QEpWDTCU0IjBECKOYvl2R8xdRtMtiM= +github.com/Azure/azure-sdk-for-go/sdk/internal v1.0.1 h1:XUNQ4mw+zJmaA2KXzP9JlQiecy1SI+Eog7xVkPiqIbg= +github.com/Azure/azure-sdk-for-go/sdk/internal v1.0.1/go.mod h1:eWRD7oawr1Mu1sLCawqVc0CUiF43ia3qQMxLscsKQ9w= +github.com/Azure/azure-sdk-for-go/sdk/storage/azblob v0.5.1 h1:BMTdr+ib5ljLa9MxTJK8x/Ds0MbBb4MfuW5BL0zMJnI= +github.com/Azure/azure-sdk-for-go/sdk/storage/azblob v0.5.1/go.mod h1:c6WvOhtmjNUWbLfOG1qxM/q0SPvQNSVJvolm+C52dIU= +github.com/Azure/go-ntlmssp v0.0.0-20220621081337-cb9428e4ac1e h1:NeAW1fUYUEWhft7pkxDf6WoUvEZJ/uOKsvtpjLnn8MU= +github.com/Azure/go-ntlmssp v0.0.0-20220621081337-cb9428e4ac1e/go.mod h1:chxPXzSsl7ZWRAuOIE23GDNzjWuZquvFlgA8xmpunjU= +github.com/AzureAD/microsoft-authentication-library-for-go v0.7.0 h1:VgSJlZH5u0k2qxSpqyghcFQKmvYckj46uymKK5XzkBM= +github.com/AzureAD/microsoft-authentication-library-for-go v0.7.0/go.mod h1:BDJ5qMFKx9DugEg3+uQSDCdbYPr5s9vBTrL9P8TpqOU= +github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= github.com/BurntSushi/toml v1.2.0 h1:Rt8g24XnyGTyglgET/PRUNlrUeu9F5L+7FilkXfZgs0= +github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo= github.com/Curtis-Milo/nat-type-identifier-go v0.0.0-20220215191915-18d42168c63d h1:62lEBImTxZ83pgzywgDNIrPPuQ+j4ep9QjqrWBn1hrU= github.com/Curtis-Milo/nat-type-identifier-go v0.0.0-20220215191915-18d42168c63d/go.mod h1:lW9x+yEjqKdPbE3+cf2fGPJXCw/hChX3Omi9QHTLFsQ= github.com/IceWhaleTech/CasaOS-Common v0.4.2-alpha3 h1:WJUYo+hJpLmza7mQngoJVeUJOfnrZevNrX5wzTuOJo0= github.com/IceWhaleTech/CasaOS-Common v0.4.2-alpha3/go.mod h1:xcemiRsXcs1zrmQxYMyExDjZ7UHYwkJqYE71IDIV0xA= +github.com/Julusian/godocdown v0.0.0-20170816220326-6d19f8ff2df8/go.mod h1:INZr5t32rG59/5xeltqoCJoNY7e5x/3xoY9WSWVWg74= +github.com/Max-Sum/base32768 v0.0.0-20191205131208-7937843c71d5 h1:w/vNc+SQRYKGWBHeDrzvvNttHwZEbSAP0kmTdORl4OI= +github.com/Max-Sum/base32768 v0.0.0-20191205131208-7937843c71d5/go.mod h1:C8yoIfvESpM3GD07OCHU7fqI7lhwyZ2Td1rbNbTAhnc= +github.com/Microsoft/go-winio v0.5.2 h1:a9IhgEQBCUEk6QCdml9CiJGhAws+YwffDHEMp1VMrpA= +github.com/Microsoft/go-winio v0.5.2/go.mod h1:WpS1mjBmmwHBEWmogvA2mj8546UReBk4v8QkMxJ6pZY= github.com/RaveNoX/go-jsoncommentstrip v1.0.0/go.mod h1:78ihd09MekBnJnxpICcwzCMzGrKSKYe4AqU6PDYYpjk= +github.com/Unknwon/goconfig v1.0.0 h1:9IAu/BYbSLQi8puFjUQApZTxIHqSwrj5d8vpP8vTq4A= +github.com/Unknwon/goconfig v1.0.0/go.mod h1:wngxua9XCNjvHjDiTiV26DaKDT+0c63QR6H5hjVUUxw= github.com/Xhofe/go-cache v0.0.0-20220723083548-714439c8af9a h1:RenIAa2q4H8UcS/cqmwdT1WCWIAH5aumP8m8RpbqVsE= github.com/Xhofe/go-cache v0.0.0-20220723083548-714439c8af9a/go.mod h1:sSBbaOg90XwWKtpT56kVujF0bIeVITnPlssLclogS04= +github.com/aalpar/deheap v0.0.0-20210914013432-0cc84d79dec3 h1:hhdWprfSpFbN7lz3W1gM40vOgvSh1WCSMxYD6gGB4Hs= +github.com/abbot/go-http-auth v0.4.0 h1:QjmvZ5gSC7jm3Zg54DqWE/T5m1t2AfDu6QlXJT0EVT0= +github.com/abbot/go-http-auth v0.4.0/go.mod h1:Cz6ARTIzApMJDzh5bRMSUou6UMSp0IEXg9km/ci7TJM= +github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= +github.com/alecthomas/template v0.0.0-20190718012654-fb15b899a751/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= +github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= +github.com/alecthomas/units v0.0.0-20190717042225-c3de453c63f4/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= +github.com/alecthomas/units v0.0.0-20190924025748-f65c72e2690d/go.mod h1:rBZYJk541a8SKzHPHnH3zbiI+7dagKZ0cgpgrD7Fyho= github.com/andybalholm/brotli v1.0.1 h1:KqhlKozYbRtJvsPrrEeXcO+N2l6NYT5A2QAFmSULpEc= github.com/andybalholm/brotli v1.0.1/go.mod h1:loMXtMfwqflxFJPmdbJO0a3KNoPuLBgiu3qAvBg8x/Y= github.com/apapsch/go-jsonmerge/v2 v2.0.0 h1:axGnT1gRIfimI7gJifB699GoE/oq+F2MU7Dml6nw9rQ= github.com/apapsch/go-jsonmerge/v2 v2.0.0/go.mod h1:lvDnEdqiQrp0O42VQGgmlKpxL1AP2+08jFMw88y4klk= +github.com/artyom/mtab v1.0.0 h1:r7OSVo5Jeqi8+LotZ0rT2kzfPIBp9KCpEJP8RQqGmSE= +github.com/artyom/mtab v1.0.0/go.mod h1:EHpkp5OmPfS1yZX+/DFTztlJ9di5UzdDLX1/XzWPXw8= +github.com/aws/aws-sdk-go v1.44.145 h1:KMVRrIyjBsNz3xGPuHIRnhIuKlb5h3Ii5e5jbi3cgnc= +github.com/aws/aws-sdk-go v1.44.145/go.mod h1:aVsgQcEevwlmQ7qHE9I3h+dtQgpqhFB+i8Phjh7fkwI= github.com/benbjohnson/clock v1.3.0 h1:ip6w0uFQkncKQ979AypyG0ER7mqUSBdKLOgAle/AT8A= github.com/benbjohnson/clock v1.3.0/go.mod h1:J11/hYXuz8f4ySSvYwY0FKfm+ezbsZBKZxNJlLklBHA= +github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q= +github.com/beorn7/perks v1.0.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8= +github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM= +github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw= github.com/bmatcuk/doublestar v1.1.1/go.mod h1:UD6OnuiIn0yFxxA2le/rnRU1G4RaI4UvFv1sNto9p6w= +github.com/buengese/sgzip v0.1.1 h1:ry+T8l1mlmiWEsDrH/YHZnCVWD2S3im1KLsyO+8ZmTU= +github.com/buengese/sgzip v0.1.1/go.mod h1:i5ZiXGF3fhV7gL1xaRRL1nDnmpNj0X061FQzOS8VMas= +github.com/calebcase/tmpfile v1.0.3 h1:BZrOWZ79gJqQ3XbAQlihYZf/YCV0H4KPIdM5K5oMpJo= +github.com/calebcase/tmpfile v1.0.3/go.mod h1:UAUc01aHeC+pudPagY/lWvt2qS9ZO5Zzof6/tIUzqeI= +github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= +github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= +github.com/cespare/xxhash/v2 v2.1.2 h1:YRXhKfTDauu4ajMg1TPgFO5jnlC2HCbmLXMcTG5cbYE= +github.com/cespare/xxhash/v2 v2.1.2/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= +github.com/chzyer/logex v1.1.10/go.mod h1:+Ywpsq7O8HXn0nuIou7OrIPyXbp3wmkHB+jjWRnGsAI= github.com/chzyer/logex v1.2.0/go.mod h1:9+9sk7u7pGNWYMkh0hdiL++6OeibzJccyQU4p4MedaY= +github.com/chzyer/readline v0.0.0-20180603132655-2972be24d48e/go.mod h1:nSuG5e5PlCu98SY8svDHJxuZscDgtXS6KTTbou5AhLI= github.com/chzyer/readline v1.5.0/go.mod h1:x22KAscuvRqlLoK9CsoYsmxoXZMMFVyOl86cAH8qUic= +github.com/chzyer/test v0.0.0-20180213035817-a1ea475d72b1/go.mod h1:Q3SI9o4m/ZMnBNeIyt5eFwwo7qiLfzFZmjNmxjkiQlU= github.com/chzyer/test v0.0.0-20210722231415-061457976a23/go.mod h1:Q3SI9o4m/ZMnBNeIyt5eFwwo7qiLfzFZmjNmxjkiQlU= +github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw= +github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc= +github.com/colinmarc/hdfs/v2 v2.3.0 h1:tMxOjXn6+7iPUlxAyup9Ha2hnmLe3Sv5DM2qqbSQ2VY= +github.com/colinmarc/hdfs/v2 v2.3.0/go.mod h1:nsyY1uyQOomU34KVQk9Qb/lDJobN1MQ/9WS6IqcVZno= +github.com/coreos/go-semver v0.3.0 h1:wkHLiw0WNATZnSG7epLsujiMCgPAc9xhjJ4tgnAxmfM= +github.com/coreos/go-semver v0.3.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk= github.com/coreos/go-systemd v0.0.0-20191104093116-d3cd4ed1dbcf h1:iW4rZ826su+pqaw19uhpSCzhj44qo35pNgKFGqzDKkU= github.com/coreos/go-systemd v0.0.0-20191104093116-d3cd4ed1dbcf/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4= github.com/coreos/go-systemd/v22 v22.3.2 h1:D9/bQk5vlXQFZ6Kwuu6zaiXJ9oTPe68++AzAJc1DzSI= github.com/coreos/go-systemd/v22 v22.3.2/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSVTIJ3seZv2GcEnc= +github.com/cpuguy83/go-md2man/v2 v2.0.2/go.mod h1:tgQtvFlXSQOSOSIRvRPT7W67SCa46tRHOmNcaadrF8o= github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= @@ -30,6 +126,9 @@ github.com/deepmap/oapi-codegen v1.12.4 h1:pPmn6qI9MuOtCz82WY2Xaw46EQjgvxednXXrP github.com/deepmap/oapi-codegen v1.12.4/go.mod h1:3lgHGMu6myQ2vqbbTXH2H1o4eXFTGnFiDaOaKKl5yas= github.com/disintegration/imaging v1.6.2 h1:w1LecBlG2Lnp8B3jk5zSuNqd7b4DXhcjwek1ei82L+c= github.com/disintegration/imaging v1.6.2/go.mod h1:44/5580QXChDfwIclfc/PCwrr44amcmDAg8hxG0Ewe4= +github.com/dnaeon/go-vcr v1.1.0 h1:ReYa/UBrRyQdant9B4fNHGoCNKw6qh6P0fsdGmZpR7c= +github.com/dropbox/dropbox-sdk-go-unofficial/v6 v6.0.5 h1:FT+t0UEDykcor4y3dMVKXIiWJETBpRgERYTGlmMd7HU= +github.com/dropbox/dropbox-sdk-go-unofficial/v6 v6.0.5/go.mod h1:rSS3kM9XMzSQ6pw91Qgd6yB5jdt70N4OdtrAf74As5M= github.com/dsnet/compress v0.0.2-0.20210315054119-f66993602bf5 h1:iFaUwBSo5Svw6L7HYpRu/0lE3e0BaElwnNO1qkNQxBY= github.com/dsnet/compress v0.0.2-0.20210315054119-f66993602bf5/go.mod h1:qssHWj60/X5sZFNxpG4HBPDHVqxNm4DfnCKgrbZOT+s= github.com/dsnet/golib v0.0.0-20171103203638-1ea166775780/go.mod h1:Lj+Z9rebOhdfkVLjJ8T6VcRQv3SXugXy999NBtR9aFY= @@ -51,6 +150,16 @@ github.com/dsoprea/go-utility/v2 v2.0.0-20221003160719-7bc88537c05e/go.mod h1:VZ github.com/dsoprea/go-utility/v2 v2.0.0-20221003172846-a3e1774ef349 h1:DilThiXje0z+3UQ5YjYiSRRzVdtamFpvBQXKwMglWqw= github.com/dsoprea/go-utility/v2 v2.0.0-20221003172846-a3e1774ef349/go.mod h1:4GC5sXji84i/p+irqghpPFZBF8tRN/Q7+700G0/DLe8= github.com/dustin/go-humanize v1.0.0/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk= +github.com/dvyukov/go-fuzz v0.0.0-20200318091601-be3528f3a813/go.mod h1:11Gm+ccJnvAhCNLlf5+cS9KjtbaD5I5zaZpFMsTHWTw= +github.com/elazarl/go-bindata-assetfs v1.0.0/go.mod h1:v+YaWX3bdea5J/mo8dSETolEo7R71Vk1u8bnjau5yw4= +github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= +github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= +github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98= +github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= +github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= +github.com/fsnotify/fsnotify v1.4.9 h1:hsms1Qyu0jgnwNXIxa+/V/PDsU6CfLf6CNO8H7IWoS4= +github.com/gabriel-vasile/mimetype v1.4.1 h1:TRWk7se+TOjCYgRth7+1/OYLNiRNIotknkFtf/dnN7Q= +github.com/gabriel-vasile/mimetype v1.4.1/go.mod h1:05Vi0w3Y9c/lNvJOdmIwvrrAhX3rYhfQQCaf9VJcv7M= github.com/geoffgarside/ber v1.1.0 h1:qTmFG4jJbwiSzSXoNJeHcOprVzZ8Ulde2Rrrifu5U9w= github.com/geoffgarside/ber v1.1.0/go.mod h1:jVPKeCbj6MvQZhwLYsGwaGI52oUorHoHKNecGT85ZCc= github.com/getkin/kin-openapi v0.113.0 h1:t9aNS/q5Agr7a55Jp1AuZ3sR2WzHESv3Dd2ys4UphsM= @@ -66,13 +175,26 @@ github.com/glebarez/go-sqlite v1.20.0 h1:6D9uRXq3Kd+W7At+hOU2eIAeahv6qcYfO8jzmvb github.com/glebarez/go-sqlite v1.20.0/go.mod h1:uTnJoqtwMQjlULmljLT73Cg7HB+2X6evsBHODyyq1ak= github.com/glebarez/sqlite v1.6.0 h1:ZpvDLv4zBi2cuuQPitRiVz/5Uh6sXa5d8eBu0xNTpAo= github.com/glebarez/sqlite v1.6.0/go.mod h1:6D6zPU/HTrFlYmVDKqBJlmQvma90P6r7sRRdkUUZOYk= +github.com/go-chi/chi/v5 v5.0.7 h1:rDTPXLDHGATaeHvVlLcR4Qe0zftYethFucbjVQ1PxU8= +github.com/go-chi/chi/v5 v5.0.7/go.mod h1:DslCQbL2OYiznFReuXYUmQ2hGd1aDpCnlMNITLSKoi8= github.com/go-errors/errors v1.0.1/go.mod h1:f4zRHt4oKfwPJE5k8C9vpYG+aDHdBFUsgrm6/TyX73Q= github.com/go-errors/errors v1.0.2/go.mod h1:psDX2osz5VnTOnFWbDeWwS7yejl+uV3FEWEp4lssFEs= github.com/go-errors/errors v1.1.1/go.mod h1:psDX2osz5VnTOnFWbDeWwS7yejl+uV3FEWEp4lssFEs= github.com/go-errors/errors v1.4.2 h1:J6MZopCL4uSllY1OfXM374weqZFFItUbrImctkmUxIA= github.com/go-errors/errors v1.4.2/go.mod h1:sIVyrIiJhuEF+Pj9Ebtd6P/rEYROXFi3BopGUQ5a5Og= +github.com/go-gl/glfw v0.0.0-20190409004039-e6da0acd62b1/go.mod h1:vR7hzQXu2zJy9AVAgeJqvqgH9Q5CA+iKCZ2gyEVpxRU= +github.com/go-gl/glfw/v3.3/glfw v0.0.0-20191125211704-12ad95a8df72/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8= +github.com/go-gl/glfw/v3.3/glfw v0.0.0-20200222043503-6f7a984d4dc4/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8= github.com/go-ini/ini v1.67.0 h1:z6ZrTEZqSWOTyH2FlglNbNgARyHG8oLW9gMELqKr06A= github.com/go-ini/ini v1.67.0/go.mod h1:ByCAeIL28uOIIG0E3PJtZPDL8WnHpFKFOtgjp+3Ies8= +github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= +github.com/go-kit/kit v0.9.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= +github.com/go-kit/log v0.1.0/go.mod h1:zbhenjAZHb184qTLMA9ZjW7ThYL0H2mk7Q6pNt4vbaY= +github.com/go-kit/log v0.2.0/go.mod h1:NwTd00d/i8cPZ3xOwwiv2PO5MOcx78fFErGNcVmBjv0= +github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9GBnD5lWE= +github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk= +github.com/go-logfmt/logfmt v0.5.0/go.mod h1:wCYkCAKZfumFQihp8CzCvQ3paCTfi41vtzG1KdI/P7A= +github.com/go-logfmt/logfmt v0.5.1/go.mod h1:WYhtIu8zTZfxdn5+rREduYbwxfcBr/Vr6KEVveWlfTs= github.com/go-ole/go-ole v1.2.6 h1:/Fpf6oFPoeFik9ty7siob0G6Ke8QvQEuVcuChpwXzpY= github.com/go-ole/go-ole v1.2.6/go.mod h1:pprOEPIfldk/42T2oK7lQ4v4JSDwmV0As9GaiUsvbm0= github.com/go-openapi/jsonpointer v0.19.5 h1:gZr+CIYByUqjcgeLXnQu2gHYQC9o73G2XUeOFYEICuY= @@ -91,6 +213,7 @@ github.com/go-playground/validator/v10 v10.11.1 h1:prmOlTVv+YjZjmRmNSF3VmspqJIxJ github.com/go-playground/validator/v10 v10.11.1/go.mod h1:i+3WkQ1FvaUjjxh1kSvIA4dMGDBiPU55YFDl0WbKdWU= github.com/go-resty/resty/v2 v2.7.0 h1:me+K9p3uhSmXtrBZ4k9jcEAfJmuC8IivWHwaLZwPrFY= github.com/go-resty/resty/v2 v2.7.0/go.mod h1:9PWDzw47qPphMRFfhsyk0NnSgvluHcljSMVIq3w7q0I= +github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY= github.com/go-test/deep v1.0.8 h1:TDsG77qcSprGbC6vTN8OuXp5g+J+b5Pcguhf7Zt61VM= github.com/go-test/deep v1.0.8/go.mod h1:5C2ZWiW0ErCdrYzpqxLbTX7MG14M9iiw8DgHncVwcsE= github.com/goccy/go-json v0.9.7/go.mod h1:6MelG93GURQebXPDq3khkgXZkazVtN9CRI+MGFi0w8I= @@ -98,8 +221,13 @@ github.com/goccy/go-json v0.9.11 h1:/pAaQDLHEoCq/5FFmSKBswWmK6H0e8g4159Kc/X/nqk= github.com/goccy/go-json v0.9.11/go.mod h1:6MelG93GURQebXPDq3khkgXZkazVtN9CRI+MGFi0w8I= github.com/godbus/dbus/v5 v5.0.4 h1:9349emZab16e7zQvpmsbtjc18ykshndd8y2PG3sgJbA= github.com/godbus/dbus/v5 v5.0.4/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA= +github.com/gofrs/flock v0.8.1 h1:+gYjHKf32LDeiEEFhQaotPbLuUXjY5ZqxKgXy7n59aw= +github.com/gofrs/flock v0.8.1/go.mod h1:F1TvTiK9OcQqauNUHlbJvyl9Qa1QvF/gOUDKA14jxHU= github.com/gofrs/uuid v4.0.0+incompatible h1:1SD/1F5pU8p29ybwgQSwpQk+mwdRrXCYuPhW6m+TnJw= github.com/gofrs/uuid v4.0.0+incompatible/go.mod h1:b2aQJv3Z4Fp6yNu3cdSllBxTCLRxnplIgP/c0N/04lM= +github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ= +github.com/gogo/protobuf v1.3.2 h1:Ov1cvc58UF3b5XjBnZv7+opcTcQFZebYjWzi34vdm4Q= +github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q= github.com/golang-jwt/jwt v3.2.2+incompatible h1:IfV12K8xAKAnZqdXVzCZ+TOjboZ2keLg81eXfW3O+oY= github.com/golang-jwt/jwt v3.2.2+incompatible/go.mod h1:8pz2t5EyA70fFQQSrl6XZXzqecmYZeUEB8OUGHkxJ+I= github.com/golang-jwt/jwt/v4 v4.4.2 h1:rcc4lwaZgFMCZ5jxF9ABolDcIHdBytAFgqFPbSJQAYs= @@ -108,10 +236,35 @@ github.com/golang/geo v0.0.0-20190916061304-5b978397cfec/go.mod h1:QZ0nwyI2jOfgR github.com/golang/geo v0.0.0-20200319012246-673a6f80352d/go.mod h1:QZ0nwyI2jOfgRAoBvP+ab5aRr7c9x7lhGEJrKvBwjWI= github.com/golang/geo v0.0.0-20210211234256-740aa86cb551 h1:gtexQ/VGyN+VVFRXSFiguSNcXmS6rkKT+X7FdIrTtfo= github.com/golang/geo v0.0.0-20210211234256-740aa86cb551/go.mod h1:QZ0nwyI2jOfgRAoBvP+ab5aRr7c9x7lhGEJrKvBwjWI= +github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= +github.com/golang/groupcache v0.0.0-20190702054246-869f871628b6/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= +github.com/golang/groupcache v0.0.0-20191227052852-215e87163ea7/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= +github.com/golang/groupcache v0.0.0-20200121045136-8c9f03a8e57e/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= +github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da h1:oI5xCqsCo564l8iNU+DwB5epxmsaqB+rhGL0m5jtYqE= +github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= +github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= +github.com/golang/mock v1.2.0/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= +github.com/golang/mock v1.3.1/go.mod h1:sBzyDLLjw3U8JLTeZvSv8jJB+tU5PVekmnlKIyFUx0Y= +github.com/golang/mock v1.4.0/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw= +github.com/golang/mock v1.4.1/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw= +github.com/golang/mock v1.4.3/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw= +github.com/golang/mock v1.4.4/go.mod h1:l3mdAwkq5BuhzHwde/uurv3sEJeZMXNpwsxVWU71h+4= github.com/golang/mock v1.6.0 h1:ErTB+efbowRARo13NNdxyJji2egdxLGQhRaY+DUumQc= github.com/golang/mock v1.6.0/go.mod h1:p6yTPP+5HYm5mzsMV8JkE6ZKdX+/wYM6Hr+LicevLPs= +github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= +github.com/golang/protobuf v1.3.3/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw= +github.com/golang/protobuf v1.3.4/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw= +github.com/golang/protobuf v1.3.5/go.mod h1:6O5/vntMXwX2lRkT1hjjk0nAC1IDOTvTlVgjlRvqsdk= +github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8= +github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA= +github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs= +github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w= +github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0= +github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8= +github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= +github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= github.com/golang/protobuf v1.5.2 h1:ROPKBNFfQgOUMifHyP+KYbvpjbdoFNs+aK7DXlji0Tw= github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= @@ -121,33 +274,97 @@ github.com/golang/snappy v0.0.4/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEW github.com/gomodule/redigo v1.8.4/go.mod h1:P9dn9mFrCBvWhGE1wpxx6fgq7BAeLBk+UUUzlpkBYO0= github.com/gomodule/redigo v1.8.9 h1:Sl3u+2BI/kk+VEatbj0scLdrFhjPmbxOc1myhDP41ws= github.com/gomodule/redigo v1.8.9/go.mod h1:7ArFNvsTjH8GMMzB4uy1snslv2BwmginuMs06a1uzZE= +github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= +github.com/google/btree v1.0.0/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= +github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M= +github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= +github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= +github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.4.1/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.1/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.2/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.3/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.4/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.6/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.9 h1:O2Tfq5qg4qc4AmwVlvv0oLiVAGB7enBSJ2x2DqQFi38= github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= github.com/google/go-github/v36 v36.0.0 h1:ndCzM616/oijwufI7nBRa+5eZHLldT+4yIB68ib5ogs= github.com/google/go-github/v36 v36.0.0/go.mod h1:LFlKC047IOqiglRGNqNb9s/iAPTnnjtlshm+bxp+kwk= -github.com/google/go-querystring v1.0.0 h1:Xkwi/a1rcvNg1PPYe5vI8GbeBY/jrVuDX5ASuANWTrk= github.com/google/go-querystring v1.0.0/go.mod h1:odCYkC5MyYFN7vkCjXpyrEuKhc/BUO6wN/zVPAxq5ck= +github.com/google/go-querystring v1.1.0 h1:AnCroh3fv4ZBgVIf1Iwtovgjaw/GiKJo8M8yD/fhyJ8= +github.com/google/go-querystring v1.1.0/go.mod h1:Kcdr2DB4koayq7X8pmAG4sNG59So17icRSOU623lUBU= github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= +github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs= +github.com/google/martian/v3 v3.0.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0= +github.com/google/pprof v0.0.0-20181206194817-3ea8567a2e57/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= +github.com/google/pprof v0.0.0-20190515194954-54271f7e092f/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= +github.com/google/pprof v0.0.0-20191218002539-d4f498aebedc/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= +github.com/google/pprof v0.0.0-20200212024743-f11f1df84d12/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= +github.com/google/pprof v0.0.0-20200229191704-1ebb73c60ed3/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= +github.com/google/pprof v0.0.0-20200430221834-fc25d7d30c6d/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= +github.com/google/pprof v0.0.0-20200708004538-1a94d8640e99/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= +github.com/google/pprof v0.0.0-20221118152302-e6195bd50e26 h1:Xim43kblpZXfIBQsbuBVKCudVG457BR2GZFIz3uw3hQ= github.com/google/pprof v0.0.0-20221118152302-e6195bd50e26/go.mod h1:dDKJzRmX4S37WGHujM7tX//fmj1uioxKzKxz3lo4HJo= +github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI= +github.com/google/uuid v1.1.2/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= github.com/google/uuid v1.3.0 h1:t6JiXgmwXMjEs8VusXIJk2BXHsn+wx8BZdTaoZ5fu7I= github.com/google/uuid v1.3.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/googleapis/enterprise-certificate-proxy v0.2.0 h1:y8Yozv7SZtlU//QXbezB6QkpuE6jMD2/gfzk4AftXjs= +github.com/googleapis/enterprise-certificate-proxy v0.2.0/go.mod h1:8C0jb7/mgJe/9KK8Lm7X9ctZC2t60YyIpYEI16jx0Qg= +github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg= +github.com/googleapis/gax-go/v2 v2.0.5/go.mod h1:DWXyrwAJ9X0FpwwEdw+IPEYBICEFu5mhpdKc/us6bOk= +github.com/googleapis/gax-go/v2 v2.7.0 h1:IcsPKeInNvYi7eqSaDjiZqDDKu5rsmunY0Y1YupQSSQ= +github.com/googleapis/gax-go/v2 v2.7.0/go.mod h1:TEop28CZZQ2y+c0VxMUmu1lV+fQx57QpBWsYpwqHJx8= github.com/googollee/go-socket.io v1.6.2 h1:olKLLHJtHz1IkL/OrTyNriZZvVQYEORNkJAqsOwPask= github.com/googollee/go-socket.io v1.6.2/go.mod h1:0vGP8/dXR9SZUMMD4+xxaGo/lohOw3YWMh2WRiWeKxg= +github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1 h1:EGx4pi6eqNxGaHF6qqu48+N2wcFQ5qg5FXgOdqsJ5d8= +github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1/go.mod h1:wJfORRmW1u3UXTncJ5qlYoELFm8eSnnEO6hX4iZ3EWY= github.com/gorilla/mux v1.8.0 h1:i40aqfkR1h2SlN9hojwV5ZA91wcXFOvkdNIeFDP5koI= github.com/gorilla/mux v1.8.0/go.mod h1:DVbg23sWSpFRCP0SfiEN6jmj59UnW/n46BH5rLB71So= +github.com/gorilla/securecookie v1.1.1 h1:miw7JPhV+b/lAHSXz4qd/nN9jRiAFV5FwjeKyCS8BvQ= +github.com/gorilla/securecookie v1.1.1/go.mod h1:ra0sb63/xPlUeL+yeDciTfxMRAA+MP+HVt/4epWDjd4= +github.com/gorilla/sessions v1.2.1 h1:DHd3rPN5lE3Ts3D8rKkQ8x/0kqfeNmBAaiSi+o7FsgI= +github.com/gorilla/sessions v1.2.1/go.mod h1:dk2InVEVJ0sfLlnXv9EAgkf6ecYs/i80K/zI+bUmuGM= github.com/gorilla/websocket v1.4.2/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE= github.com/gorilla/websocket v1.5.0 h1:PPwGk2jz7EePpoHN/+ClbZu8SPxiqlu12wZP/3sWmnc= github.com/gorilla/websocket v1.5.0/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE= github.com/h2non/filetype v1.1.3 h1:FKkx9QbD7HR/zjK1Ia5XiBsq9zdLi5Kf3zGyFTAFkGg= github.com/h2non/filetype v1.1.3/go.mod h1:319b3zT68BvV+WRj7cwy856M2ehB3HqNOt6sy1HndBY= +github.com/hanwen/go-fuse/v2 v2.1.0 h1:+32ffteETaLYClUj0a3aHjZ1hOPxxaNEHiZiujuDaek= +github.com/hashicorp/errwrap v1.0.0 h1:hLrqtEDnRye3+sgx6z4qVLNuviH3MR5aQ0ykNJa/UYA= +github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4= +github.com/hashicorp/go-multierror v1.1.1 h1:H5DkEtf6CXdFp0N0Em5UCwQpXMWke8IA0+lD48awMYo= +github.com/hashicorp/go-multierror v1.1.1/go.mod h1:iw975J/qwKPdAO1clOe2L8331t/9/fmwbPZ6JB6eMoM= +github.com/hashicorp/go-uuid v1.0.2/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= +github.com/hashicorp/go-uuid v1.0.3 h1:2gKiV6YVmrJ1i2CKKa9obLvRieoRGviZFL26PcT/Co8= +github.com/hashicorp/go-uuid v1.0.3/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= +github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= +github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= github.com/hirochachacha/go-smb2 v1.1.0 h1:b6hs9qKIql9eVXAiN0M2wSFY5xnhbHAQoCwRKbaRTZI= github.com/hirochachacha/go-smb2 v1.1.0/go.mod h1:8F1A4d5EZzrGu5R7PU163UcMRDJQl4FtcxjBfsY8TZE= +github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= +github.com/ianlancetaylor/demangle v0.0.0-20181102032728-5e5cf60278f6/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= github.com/ianlancetaylor/demangle v0.0.0-20220319035150-800ac71e25c2/go.mod h1:aYm2/VgdVmcIU8iMfdMvDMsRAQjcfZSKFby6HOFvi/w= +github.com/iguanesolutions/go-systemd/v5 v5.1.0 h1:UWprhbpxjLM0vvwu4MxaBR+/KzSxgvnKpM9Q3MBhTAc= +github.com/iguanesolutions/go-systemd/v5 v5.1.0/go.mod h1:XprNDEZ9zdPzEg1WrmpV1BnGorgP0WP40AGurMxeQOY= +github.com/inconshreveable/mousetrap v1.0.1 h1:U3uMjPSQEBMNp1lFxmllqCPM6P5u/Xq7Pgzkat/bFNc= +github.com/inconshreveable/mousetrap v1.0.1/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLfsEA9PFc4w1p2J65bw= github.com/invopop/yaml v0.1.0 h1:YW3WGUoJEXYfzWBjn00zIlrw7brGVD0fUKRYDPAPhrc= github.com/invopop/yaml v0.1.0/go.mod h1:2XuRLgs/ouIrW3XNzuNj7J3Nvu/Dig5MXvbCEdiBN3Q= +github.com/jcmturner/aescts/v2 v2.0.0 h1:9YKLH6ey7H4eDBXW8khjYslgyqG2xZikXP0EQFKrle8= +github.com/jcmturner/aescts/v2 v2.0.0/go.mod h1:AiaICIRyfYg35RUkr8yESTqvSy7csK90qZ5xfvvsoNs= +github.com/jcmturner/dnsutils/v2 v2.0.0 h1:lltnkeZGL0wILNvrNiVCR6Ro5PGU/SeBvVO/8c/iPbo= +github.com/jcmturner/dnsutils/v2 v2.0.0/go.mod h1:b0TnjGOvI/n42bZa+hmXL+kFJZsFT7G4t3HTlQ184QM= +github.com/jcmturner/gofork v1.7.6 h1:QH0l3hzAU1tfT3rZCnW5zXl+orbkNMMRGJfdJjHVETg= +github.com/jcmturner/gofork v1.7.6/go.mod h1:1622LH6i/EZqLloHfE7IeZ0uEJwMSUyQ/nDd82IeqRo= +github.com/jcmturner/goidentity/v6 v6.0.1 h1:VKnZd2oEIMorCTsFBnJWbExfNN7yZr3EhJAxwOkZg6o= +github.com/jcmturner/goidentity/v6 v6.0.1/go.mod h1:X1YW3bgtvwAXju7V3LCIMpY0Gbxyjn/mY9zx4tFonSg= +github.com/jcmturner/gokrb5/v8 v8.4.3 h1:iTonLeSJOn7MVUtyMT+arAn5AKAPrkilzhGw8wE/Tq8= +github.com/jcmturner/gokrb5/v8 v8.4.3/go.mod h1:dqRwJGXznQrzw6cWmyo6kH+E7jksEQG/CyVWsJEsJO0= +github.com/jcmturner/rpc/v2 v2.0.3 h1:7FXXj8Ti1IaVFpSAziCZWNzbNuZmnvw/i6CqLNdWfZY= +github.com/jcmturner/rpc/v2 v2.0.3/go.mod h1:VUJYCIDm3PVOEHw8sgt091/20OJjskO/YJki3ELg/Hc= github.com/jessevdk/go-flags v1.4.0/go.mod h1:4FA24M0QyGHXBuZZK/XkWh8h0e1EYbRYJSGM75WSRxI= github.com/jessevdk/go-flags v1.5.0/go.mod h1:Fw0T6WPc1dYxT4mKEZRfG5kJhaTDP9pj1c2EWnYs/m4= github.com/jinzhu/inflection v1.0.0 h1:K317FqzuhWc8YvSVlFMCCUb36O/S9MCKRDI7QkRKD/E= @@ -155,12 +372,30 @@ github.com/jinzhu/inflection v1.0.0/go.mod h1:h+uFLlag+Qp1Va5pdKtLDYj+kHp5pxUVkr github.com/jinzhu/now v1.1.4/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/z8= github.com/jinzhu/now v1.1.5 h1:/o9tlHleP7gOFmsnYNz3RGnqzefHA47wQpKrrdTIwXQ= github.com/jinzhu/now v1.1.5/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/z8= +github.com/jmespath/go-jmespath v0.4.0 h1:BEgLn5cpjn8UN1mAw4NjwDrS35OdebyEtFe+9YPoQUg= +github.com/jmespath/go-jmespath v0.4.0/go.mod h1:T8mJZnbsbmF+m6zOOFylbeCJqk5+pHWvzYPziyZiYoo= +github.com/jmespath/go-jmespath/internal/testify v1.5.1 h1:shLQSRRSCCPj3f2gpwzGwWFoC7ycTf1rcQZHOlsJ6N8= +github.com/jmespath/go-jmespath/internal/testify v1.5.1/go.mod h1:L3OGu8Wl2/fWfCI6z80xFu9LTZmf1ZRjMHUOPmWr69U= github.com/josharian/intern v1.0.0 h1:vlS4z54oSdjm0bgjRigI+G1HpF+tI+9rE5LLzOg8HmY= github.com/josharian/intern v1.0.0/go.mod h1:5DoeVV0s6jJacbCEi61lwdGj/aVlrQvzHFFd8Hwg//Y= +github.com/jpillora/backoff v1.0.0/go.mod h1:J/6gKK9jxlEcS3zixgDgUAsiuZ7yrSoa/FX5e0EB2j4= +github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU= +github.com/json-iterator/go v1.1.10/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= +github.com/json-iterator/go v1.1.11/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM= github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo= +github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU= +github.com/jstemmer/go-junit-report v0.9.1/go.mod h1:Brl9GWCQeLvo8nXZwPNNblvFj/XSXhF0NWZEnDohbsk= +github.com/jtolds/gls v4.20.0+incompatible h1:xdiiI2gbIgH/gLH7ADydsJ1uDOEzR8yvV7C0MuV77Wo= +github.com/jtolds/gls v4.20.0+incompatible/go.mod h1:QJZ7F/aHp+rZTRtaJ1ow/lLfFfVYBRgL+9YlvaHOwJU= github.com/juju/gnuflag v0.0.0-20171113085948-2ce1bb71843d/go.mod h1:2PavIy+JPciBPrBUjwbNvtwB6RQlve+hkpll6QSNmOE= +github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w= +github.com/julienschmidt/httprouter v1.3.0/go.mod h1:JR6WtHb+2LUe8TCKY3cZOxFyyO8IZAc4RVcycCCAKdM= +github.com/jzelinskie/whirlpool v0.0.0-20201016144138-0675e54bb004 h1:G+9t9cEtnC9jFiTxyptEKuNIAbiN5ZCQzX2a74lj3xg= +github.com/jzelinskie/whirlpool v0.0.0-20201016144138-0675e54bb004/go.mod h1:KmHnJWQrgEvbuy0vcvj00gtMqbvNn1L+3YUZLK/B92c= github.com/kballard/go-shellquote v0.0.0-20180428030007-95032a82bc51/go.mod h1:CzGEWj7cYgsdH8dAjBGEr58BoE7ScuLd+fwFZ44+/x8= +github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8= +github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= github.com/klauspost/compress v1.4.1/go.mod h1:RyIbtBH6LamlWaDj8nUwkbUhJ87Yi3uG0guNDohfE1A= github.com/klauspost/compress v1.11.4/go.mod h1:aoV0uJVorq1K+umq18yTdKaF57EivdYsUV+/s2qKfXs= github.com/klauspost/compress v1.15.13 h1:NFn1Wr8cfnenSJSA46lLq4wHCcBzKTSjnBIexDMMOV0= @@ -168,6 +403,15 @@ github.com/klauspost/compress v1.15.13/go.mod h1:QPwzmACJjUTFsnSHH934V6woptycfrD github.com/klauspost/cpuid v1.2.0/go.mod h1:Pj4uuM528wm8OyEC2QMXAi2YiTZ96dNQPGgoMS4s3ek= github.com/klauspost/pgzip v1.2.5 h1:qnWYvvKqedOF2ulHpMG72XQol4ILEJ8k2wwRl/Km8oE= github.com/klauspost/pgzip v1.2.5/go.mod h1:Ch1tH69qFZu15pkjo5kYi6mth2Zzwzt50oCQKQE9RUs= +github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= +github.com/konsorten/go-windows-terminal-sequences v1.0.3/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= +github.com/koofr/go-httpclient v0.0.0-20200420163713-93aa7c75b348 h1:Lrn8srO9JDBCf2iPjqy62stl49UDwoOxZ9/NGVi+fnk= +github.com/koofr/go-httpclient v0.0.0-20200420163713-93aa7c75b348/go.mod h1:JBLy//Q5jzU3XSMxdONTD5EIj1LhTPktosxG2Bw1iho= +github.com/koofr/go-koofrclient v0.0.0-20190724113126-8e5366da203a h1:02cx9xF4W2FQ1oh8CK9dWV5BnZK2mUtcbr9xR+bZiKk= +github.com/koofr/go-koofrclient v0.0.0-20190724113126-8e5366da203a/go.mod h1:MRAz4Gsxd+OzrZ0owwrUHc0zLESL+1Y5syqK/sJxK2A= +github.com/kr/fs v0.1.0 h1:Jskdu9ieNAYnjxsi0LbQp1ulIKZV1LAFgK1tWhpZgl8= +github.com/kr/fs v0.1.0/go.mod h1:FFnZGqtBN9Gxj7eW1uZ42v5BccTP0vu6NEaFoC2HwRg= +github.com/kr/logfmt v0.0.0-20140226030751-b84e30acd515/go.mod h1:+0opPa2QZZtGFBFZlji/RkVcI2GknAs/DXo4wKdlNEc= github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= github.com/kr/pretty v0.3.0 h1:WgNl7dwNpEZ6jJ9k1snq4pZsg7DOEN8hP9Xw0Tsjwk0= @@ -176,6 +420,8 @@ github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= +github.com/kylelemons/godebug v1.1.0 h1:RPNrshWIDI6G2gRW9EHilWtl7Z6Sb1BR0xunSBf0SNc= +github.com/kylelemons/godebug v1.1.0/go.mod h1:9/0rRGxNHcop5bhtWyNeEfOS8JIWk580+fNqagV/RAw= github.com/labstack/echo/v4 v4.10.0 h1:5CiyngihEO4HXsz3vVsJn7f8xAlWwRr3aY6Ih280ZKA= github.com/labstack/echo/v4 v4.10.0/go.mod h1:S/T/5fy/GigaXnHTkh0ZGe4LpkkQysvRjFMSUTkDRNQ= github.com/labstack/gommon v0.4.0 h1:y7cvthEAEbU0yHOf4axH8ZG2NH8knB9iNSoTO8dyIk8= @@ -199,8 +445,13 @@ github.com/mattn/go-isatty v0.0.16 h1:bq3VjFmv/sOjHtdEhmkEV4x1AJtvUvOJ2PFAZ5+peK github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM= github.com/mattn/go-sqlite3 v1.14.15 h1:vfoHhTN1af61xCRSWzFIWzx2YskyMTwHLrExkBOjvxI= github.com/mattn/go-sqlite3 v1.14.15/go.mod h1:2eHXhiwb8IkHr+BDWZGa96P6+rkvnG63S2DGjv9HUNg= +github.com/matttproud/golang_protobuf_extensions v1.0.1 h1:4hp9jkHxhMHkqkrB3Ix0jegS5sx/RkqARlsWZ6pIwiU= +github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0= github.com/mholt/archiver/v3 v3.5.1 h1:rDjOBX9JSF5BvoJGvjqK479aL70qh9DIpZCl+k7Clwo= github.com/mholt/archiver/v3 v3.5.1/go.mod h1:e3dqJ7H78uzsRSEACH1joayhuSyhnonssnDhppzS1L4= +github.com/miekg/dns v1.1.42/go.mod h1:+evo5L0630/F6ca/Z9+GAqzhjGyn8/c+TBaOyfEl0V4= +github.com/mitchellh/go-homedir v1.1.0 h1:lukF9ziXFxDFPkA1vsr5zpc1XuPDn/wFntq5mG+4E0Y= +github.com/mitchellh/go-homedir v1.1.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= github.com/moby/sys/mount v0.3.3 h1:fX1SVkXFJ47XWDoeFW4Sq7PdQJnV2QIDZAqjNqgEjUs= github.com/moby/sys/mount v0.3.3/go.mod h1:PBaEorSNTLG5t/+4EgukEQVlAvVEc6ZjTySwKdqp5K0= github.com/moby/sys/mountinfo v0.6.2 h1:BzJjoreD5BMFNmD9Rus6gdd1pLuecOFPt8wC+Vygl78= @@ -208,47 +459,135 @@ github.com/moby/sys/mountinfo v0.6.2/go.mod h1:IJb6JQeOklcdMU9F5xQ8ZALD+CUr5VlGp github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w8PVh93nsPXa1VrQ6jlwL5oN8l14QlcNfg= github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= +github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= +github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M= github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk= github.com/mohae/deepcopy v0.0.0-20170929034955-c48cc78d4826 h1:RWengNIwukTxcDr9M+97sNutRR1RKhG96O6jWumTTnw= github.com/mohae/deepcopy v0.0.0-20170929034955-c48cc78d4826/go.mod h1:TaXosZuwdSHYgviHp1DAtfrULt5eUgsSMsZf+YrPgl8= +github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= +github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= +github.com/ncw/go-acd v0.0.0-20201019170801-fe55f33415b1 h1:nAjWYc03awJAjsozNehdGZsm5LP7AhLOvjgbS8zN1tk= +github.com/ncw/go-acd v0.0.0-20201019170801-fe55f33415b1/go.mod h1:MLIrzg7gp/kzVBxRE1olT7CWYMCklcUWU+ekoxOD9x0= +github.com/ncw/swift/v2 v2.0.1 h1:q1IN8hNViXEv8Zvg3Xdis4a3c4IlIGezkYz09zQL5J0= +github.com/ncw/swift/v2 v2.0.1/go.mod h1:z0A9RVdYPjNjXVo2pDOPxZ4eu3oarO1P91fTItcb+Kg= github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e/go.mod h1:zD1mROLANZcx1PVRCS0qkT7pwLkGfwJo4zjcN/Tysno= github.com/nwaples/rardecode v1.1.0 h1:vSxaY8vQhOcVr4mm5e8XllHWTiM4JF507A0Katqw7MQ= github.com/nwaples/rardecode v1.1.0/go.mod h1:5DzqNKiOdpKKBH87u8VlvAnPZMXcGRhxWkRpHbbfGS0= +github.com/nxadm/tail v1.4.8 h1:nPr65rt6Y5JFSKQO7qToXr7pePgD6Gwiw05lkbyAQTE= +github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= +github.com/onsi/ginkgo v1.12.0/go.mod h1:oUhWkIvk5aDxtKvDDuw8gItl8pKl42LzjC9KZE0HfGg= +github.com/onsi/ginkgo v1.16.4 h1:29JGrr5oVBm5ulCWet69zQkzWipVXIol6ygQUe/EzNc= +github.com/onsi/gomega v1.7.1/go.mod h1:XdKZgCCFLUoM/7CFJVPcG8C1xQ1AJ0vpAezJrB7JYyY= +github.com/onsi/gomega v1.9.0/go.mod h1:Ho0h+IUsWyvy1OpqCwxlQ/21gkhVunqlU8fDGcoTdcA= +github.com/onsi/gomega v1.13.0 h1:7lLHu94wT9Ij0o6EWWclhu0aOh32VxhkwEJvzuWPeak= +github.com/oracle/oci-go-sdk/v65 v65.26.1 h1:Ms20RSRj+CuvQmw5ET1TkmzxLBI+bmLjZ6NYANA3gkk= +github.com/oracle/oci-go-sdk/v65 v65.26.1/go.mod h1:oyMrMa1vOzzKTmPN+kqrTR9y9kPA2tU1igN3NUSNTIE= github.com/patrickmn/go-cache v2.1.0+incompatible h1:HRMgzkcYKYpi3C8ajMPV8OFXaaRUnok+kx1WdO15EQc= github.com/patrickmn/go-cache v2.1.0+incompatible/go.mod h1:3Qf8kWWT7OJRJbdiICTKqZju1ZixQ/KpMGzzAfe6+WQ= github.com/pelletier/go-toml/v2 v2.0.1/go.mod h1:r9LEWfGN8R5k0VXJ+0BkIe7MYkRdwZOjgMj2KwnJFUo= github.com/pelletier/go-toml/v2 v2.0.6 h1:nrzqCb7j9cDFj2coyLNLaZuJTLjWjlaz6nvTvIwycIU= github.com/pelletier/go-toml/v2 v2.0.6/go.mod h1:eumQOmlWiOPt5WriQQqoM5y18pDHwha2N+QD+EUNTek= +github.com/pengsrc/go-shared v0.2.1-0.20190131101655-1999055a4a14 h1:XeOYlK9W1uCmhjJSsY78Mcuh7MVkNjTzmHx1yBzizSU= +github.com/pengsrc/go-shared v0.2.1-0.20190131101655-1999055a4a14/go.mod h1:jVblp62SafmidSkvWrXyxAme3gaTfEtWwRPGz5cpvHg= github.com/perimeterx/marshmallow v1.1.4 h1:pZLDH9RjlLGGorbXhcaQLhfuV0pFMNfPO55FuFkxqLw= github.com/perimeterx/marshmallow v1.1.4/go.mod h1:dsXbUu8CRzfYP5a87xpp0xq9S3u0Vchtcl8we9tYaXw= github.com/pierrec/lz4/v4 v4.1.2 h1:qvY3YFXRQE/XB8MlLzJH7mSzBs74eA2gg52YTk6jUPM= github.com/pierrec/lz4/v4 v4.1.2/go.mod h1:gZWDp/Ze/IJXGXf23ltt2EXimqmTUXEy0GFuRQyBid4= +github.com/pkg/browser v0.0.0-20210115035449-ce105d075bb4 h1:Qj1ukM4GlMWXNdMBuXcXfz/Kw9s1qm0CLY32QxuSImI= +github.com/pkg/browser v0.0.0-20210115035449-ce105d075bb4/go.mod h1:N6UoU20jOqggOuDwUaBQpluzLNDqif3kq9z2wpdYEfQ= github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsKMEsNbx1VGcRFpLqf3715MtcvvzbA= +github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= +github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= +github.com/pkg/sftp v1.13.5 h1:a3RLUqkyjYRtBTZJZ1VRrKbN3zhuPLlUc3sphVz81go= +github.com/pkg/sftp v1.13.5/go.mod h1:wHDZ0IZX6JcBYRK1TH9bcVq8G7TLpVHYIGJRFnmPfxg= +github.com/pkg/xattr v0.4.9 h1:5883YPCtkSd8LFbs13nXplj9g9tlrwoJRjgpgMu1/fE= +github.com/pkg/xattr v0.4.9/go.mod h1:di8WF84zAKk8jzR1UBTEWh9AUlIZZ7M/JNt8e9B6ktU= github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= github.com/power-devops/perfstat v0.0.0-20210106213030-5aafc221ea8c h1:ncq/mPwQF4JjgDlrVEn3C11VoGHZN7m8qihwgMEtzYw= github.com/power-devops/perfstat v0.0.0-20210106213030-5aafc221ea8c/go.mod h1:OmDBASR4679mdNQnz2pUhc2G8CO2JrUAVFDRBDP/hJE= +github.com/prometheus/client_golang v0.9.1/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw= +github.com/prometheus/client_golang v1.0.0/go.mod h1:db9x61etRT2tGnBNRi70OPL5FsnadC4Ky3P0J6CfImo= +github.com/prometheus/client_golang v1.7.1/go.mod h1:PY5Wy2awLA44sXw4AOSfFBetzPP4j5+D6mVACh+pe2M= +github.com/prometheus/client_golang v1.11.0/go.mod h1:Z6t4BnS23TR94PD6BsDNk8yVqroYurpAkEiz0P2BEV0= +github.com/prometheus/client_golang v1.12.1/go.mod h1:3Z9XVyYiZYEO+YQWt3RD2R3jrbd179Rt297l4aS6nDY= +github.com/prometheus/client_golang v1.14.0 h1:nJdhIvne2eSX/XRAFV9PcvFFRbrjbcTUj0VP62TMhnw= +github.com/prometheus/client_golang v1.14.0/go.mod h1:8vpkKitgIVNcqrRBWh1C4TIUQgYNtG/XQE4E/Zae36Y= +github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo= +github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= +github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= +github.com/prometheus/client_model v0.2.0/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= +github.com/prometheus/client_model v0.3.0 h1:UBgGFHqYdG/TPFD1B1ogZywDqEkwp3fBMvqdiQ7Xew4= +github.com/prometheus/client_model v0.3.0/go.mod h1:LDGWKZIo7rky3hgvBe+caln+Dr3dPggB5dvjtD7w9+w= +github.com/prometheus/common v0.4.1/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4= +github.com/prometheus/common v0.10.0/go.mod h1:Tlit/dnDKsSWFlCLTWaA1cyBgKHSMdTB80sz/V91rCo= +github.com/prometheus/common v0.26.0/go.mod h1:M7rCNAaPfAosfx8veZJCuw84e35h3Cfd9VFqTh1DIvc= +github.com/prometheus/common v0.32.1/go.mod h1:vu+V0TpY+O6vW9J44gczi3Ap/oXXR10b+M/gUGO4Hls= +github.com/prometheus/common v0.37.0 h1:ccBbHCgIiT9uSoFY0vX8H3zsNR5eLt17/RQLUvn8pXE= +github.com/prometheus/common v0.37.0/go.mod h1:phzohg0JFMnBEFGxTDbfu3QyL5GI8gTQJFhYO5B3mfA= +github.com/prometheus/procfs v0.0.0-20181005140218-185b4288413d/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= +github.com/prometheus/procfs v0.0.2/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA= +github.com/prometheus/procfs v0.1.3/go.mod h1:lV6e/gmhEcM9IjHGsFOCxxuZ+z1YqCvr4OA4YeYWdaU= +github.com/prometheus/procfs v0.6.0/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA= +github.com/prometheus/procfs v0.7.3/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA= +github.com/prometheus/procfs v0.8.0 h1:ODq8ZFEaYeCaZOJlZZdJA2AbQR98dSHSM1KW/You5mo= +github.com/prometheus/procfs v0.8.0/go.mod h1:z7EfXMXOkbkqb9IINtpCn86r/to3BnA0uaxHdg830/4= +github.com/putdotio/go-putio/putio v0.0.0-20200123120452-16d982cac2b8 h1:Y258uzXU/potCYnQd1r6wlAnoMB68BiCkCcCnKx1SH8= +github.com/putdotio/go-putio/putio v0.0.0-20200123120452-16d982cac2b8/go.mod h1:bSJjRokAHHOhA+XFxplld8w2R/dXLH7Z3BZ532vhFwU= +github.com/rclone/ftp v0.0.0-20221014110213-e44dedbc76c6 h1:J832KfU2Z44Ck3XR5bvw2UxShP0QnjueruNQ6dTYH+g= +github.com/rclone/ftp v0.0.0-20221014110213-e44dedbc76c6/go.mod h1:qRpxqlna6CaIq9fSRud1bDC5S7EEUEou0j8nMZ0lxO8= +github.com/rclone/rclone v1.61.1 h1:bFqzxirefljtiBoDO0BXFvC81igHjJE44KAOxClSONA= +github.com/rclone/rclone v1.61.1/go.mod h1:DAJ02MbKrfAjM5VCwDpIvb4Q8aooru5uBDN9Lmruwg0= github.com/remyoudompheng/bigfft v0.0.0-20200410134404-eec4a21b6bb0 h1:OdAsTTz6OkFY5QxjkYwrChwuRruF69c169dPK26NUlk= github.com/remyoudompheng/bigfft v0.0.0-20200410134404-eec4a21b6bb0/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo= +github.com/rfjakob/eme v1.1.2 h1:SxziR8msSOElPayZNFfQw4Tjx/Sbaeeh3eRvrHVMUs4= +github.com/rfjakob/eme v1.1.2/go.mod h1:cVvpasglm/G3ngEfcfT/Wt0GwhkuO32pf/poW6Nyk1k= +github.com/robertkrimen/godocdown v0.0.0-20130622164427-0bfa04905481/go.mod h1:C9WhFzY47SzYBIvzFqSvHIR6ROgDo4TtdTuRaOMjF/s= github.com/robfig/cron v1.2.0 h1:ZjScXvvxeQ63Dbyxy76Fj3AT3Ut0aKsyd2/tl3DTMuQ= github.com/robfig/cron v1.2.0/go.mod h1:JGuDeoQd7Z6yL4zQhZ3OPEVHB7fL6Ka6skscFHfmt2k= +github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= github.com/rogpeppe/go-internal v1.6.1/go.mod h1:xXDCJY+GAPziupqXw64V24skbSoqbTEfhy4qGm1nDQc= github.com/rogpeppe/go-internal v1.8.0 h1:FCbCCtXNOY3UtUuHUYaghJg4y7Fd14rXifAYUAtL9R8= github.com/rogpeppe/go-internal v1.8.0/go.mod h1:WmiCO8CzOY8rg0OYDC4/i/2WRWAB6poM+XZ2dLUbcbE= +github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= github.com/satori/go.uuid v1.2.0 h1:0uYX9dsZ2yD7q2RtLRtPSdGDWzjeM3TbMJP9utgA0ww= github.com/satori/go.uuid v1.2.0/go.mod h1:dA0hQrYB0VpLJoorglMZABFdXlWrHn1NEOzdhQKdks0= github.com/shirou/gopsutil/v3 v3.22.11 h1:kxsPKS+Eeo+VnEQ2XCaGJepeP6KY53QoRTETx3+1ndM= github.com/shirou/gopsutil/v3 v3.22.11/go.mod h1:xl0EeL4vXJ+hQMAGN8B9VFpxukEMA0XdevQOe5MZ1oY= +github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo= +github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE= +github.com/sirupsen/logrus v1.6.0/go.mod h1:7uNnSEd1DgxDLC74fIahvMZmmYsHGZGEOFrfsX/uA88= +github.com/sirupsen/logrus v1.7.0/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0= github.com/sirupsen/logrus v1.9.0 h1:trlNQbNUG3OdDrDil03MCb1H2o9nJ1x4/5LYw7byDE0= github.com/sirupsen/logrus v1.9.0/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ= +github.com/skratchdot/open-golang v0.0.0-20200116055534-eef842397966 h1:JIAuq3EEf9cgbU6AtGPK4CTG3Zf6CKMNqf0MHTggAUA= +github.com/skratchdot/open-golang v0.0.0-20200116055534-eef842397966/go.mod h1:sUM3LWHvSMaG192sy56D9F7CNvL7jUJVXoqM1QKLnog= +github.com/smartystreets/assertions v1.2.0 h1:42S6lae5dvLc7BrLu/0ugRtcFVjoJNMC/N3yZFZkDFs= +github.com/smartystreets/assertions v1.2.0/go.mod h1:tcbTF8ujkAEcZ8TElKY+i30BzYlVhC/LOxJk7iOWnoo= +github.com/smartystreets/goconvey v1.7.2 h1:9RBaZCeXEQ3UselpuwUQHltGVXvdwm6cv1hgR6gDIPg= +github.com/smartystreets/goconvey v1.7.2/go.mod h1:Vw0tHAZW6lzCRk3xgdin6fKYcG+G3Pg9vgXWeJpQFMM= +github.com/sony/gobreaker v0.5.0 h1:dRCvqm0P490vZPmy7ppEk2qCnCieBooFJ+YoXGYB+yg= +github.com/sony/gobreaker v0.5.0/go.mod h1:ZKptC7FHNvhBz7dN2LGjPVBz2sZJmc0/PkyDJOjmxWY= +github.com/spacemonkeygo/monkit/v3 v3.0.17 h1:rqIuLhRUr2UtS3WNVbPY/BwvjlwKVvSOVY5p0QVocxE= +github.com/spacemonkeygo/monkit/v3 v3.0.17/go.mod h1:kj1ViJhlyADa7DiA4xVnTuPA46lFKbM7mxQTrXCuJP4= +github.com/spf13/cobra v1.6.1 h1:o94oiPyS4KD1mPy2fmcYYHHfCxLqYjJOhGsCHFZtEzA= +github.com/spf13/cobra v1.6.1/go.mod h1:IOw/AERYS7UzyrGinqmz6HLUo219MORXGxhbaJUqzrY= +github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA= +github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= github.com/spkg/bom v0.0.0-20160624110644-59b7046e48ad/go.mod h1:qLr4V1qq6nMqFKkMo8ZTx3f+BZEkzsRUY10Xsm2mwU0= +github.com/stephens2424/writerset v1.0.2/go.mod h1:aS2JhsMn6eA7e82oNmW4rfsgAOp9COBTTl8mzkwADnc= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= +github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= +github.com/stretchr/objx v0.5.0 h1:1zr/of2m5FGMsad5YfcqgdqdWrIhu+EBEJRhR1U7z/c= github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= +github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= +github.com/stretchr/testify v1.3.1-0.20190311161405-34c6fa2dc709/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= +github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= @@ -256,6 +595,8 @@ github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/ github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= github.com/stretchr/testify v1.8.1 h1:w7B6lhMri9wdJUVmEZPGGhZzrYTPvgJArz7wNPgYKsk= github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= +github.com/t3rm1n4l/go-mega v0.0.0-20220725095014-c4e0c2b5debf h1:Y43S3e9P1NPs/QF4R5/SdlXj2d31540hP4Gk8VKNvDg= +github.com/t3rm1n4l/go-mega v0.0.0-20220725095014-c4e0c2b5debf/go.mod h1:c+cGNU1qi9bO7ZF4IRMYk+KaZTNiQ/gQrSbyMmGFq1Q= github.com/tidwall/gjson v1.14.4 h1:uo0p8EbA09J7RQaflQ1aBRffTR7xedD2bcIVSYxLnkM= github.com/tidwall/gjson v1.14.4/go.mod h1:/wbyibRr2FHMks5tjHJ5F8dMZh3AcwJEMf5vlfC0lxk= github.com/tidwall/match v1.1.1 h1:+Ho715JplO36QYgwN9PGYNhgZvoUSc9X2c80KVTi+GA= @@ -266,6 +607,8 @@ github.com/tklauser/go-sysconf v0.3.11 h1:89WgdJhk5SNwJfu+GKyYveZ4IaJ7xAkecBo+Kd github.com/tklauser/go-sysconf v0.3.11/go.mod h1:GqXfhXY3kiPa0nAXPDIQIWzJbMCB7AmcWpGR8lSZfqI= github.com/tklauser/numcpus v0.6.0 h1:kebhY2Qt+3U6RNK7UqpYNA+tJ23IBEGKkB7JQBfDYms= github.com/tklauser/numcpus v0.6.0/go.mod h1:FEZLMke0lhOUG6w2JadTzp0a+Nl8PF/GFkQ5UVIcaL4= +github.com/tv42/httpunix v0.0.0-20191220191345-2ba4b9c3382c h1:u6SKchux2yDvFQnDHS3lPnIRmfVJ5Sxy3ao2SIdysLQ= +github.com/tv42/httpunix v0.0.0-20191220191345-2ba4b9c3382c/go.mod h1:hzIxponao9Kjc7aWznkXaL4U4TWaDSs8zcsY4Ka08nM= github.com/ugorji/go v1.2.7/go.mod h1:nF9osbDWLy6bDVv/Rtoh6QgnvNDpmCalQV5urGCCS6M= github.com/ugorji/go/codec v1.2.7 h1:YPXUKf7fYbp/y8xloBqZOw2qaVggbfwMlI8WM3wZUJ0= github.com/ugorji/go/codec v1.2.7/go.mod h1:WGN1fab3R1fzQlVQTkfxVtIBhWDRqOviHU95kRgeqEY= @@ -277,12 +620,37 @@ github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyC github.com/valyala/fasttemplate v1.2.1/go.mod h1:KHLXt3tVN2HBp8eijSv/kGJopbvo7S+qRAEEKiv+SiQ= github.com/valyala/fasttemplate v1.2.2 h1:lxLXG0uE3Qnshl9QyaK6XJxMXlQZELvChBOCmQD0Loo= github.com/valyala/fasttemplate v1.2.2/go.mod h1:KHLXt3tVN2HBp8eijSv/kGJopbvo7S+qRAEEKiv+SiQ= +github.com/vivint/infectious v0.0.0-20200605153912-25a574ae18a3 h1:zMsHhfK9+Wdl1F7sIKLyx3wrOFofpb3rWFbA4HgcK5k= +github.com/vivint/infectious v0.0.0-20200605153912-25a574ae18a3/go.mod h1:R0Gbuw7ElaGSLOZUSwBm/GgVwMd30jWxBDdAyMOeTuc= +github.com/winfsp/cgofuse v1.5.1-0.20221118130120-84c0898ad2e0 h1:j3un8DqYvvAOqKI5OPz+/RRVhDFipbPKI4t2Uk5RBJw= +github.com/xanzy/ssh-agent v0.3.3 h1:+/15pJfg/RsTxqYcX6fHqOXZwwMP+2VyYWJeWM2qQFM= +github.com/xanzy/ssh-agent v0.3.3/go.mod h1:6dzNDKs0J9rVPHPhaGCukekBHKqfl+L3KghI1Bc68Uw= github.com/xi2/xz v0.0.0-20171230120015-48954b6210f8 h1:nIPpBwaJSVYIxUFsDv3M8ofmx9yWTog9BfvIu0q41lo= github.com/xi2/xz v0.0.0-20171230120015-48954b6210f8/go.mod h1:HUYIGzjTL3rfEspMxjDjgmT5uz5wzYJKVo23qUhYTos= +github.com/youmark/pkcs8 v0.0.0-20201027041543-1326539a0a0a h1:fZHgsYlfvtyqToslyjUt3VOPF4J7aK/3MPcK7xp3PDk= +github.com/youmark/pkcs8 v0.0.0-20201027041543-1326539a0a0a/go.mod h1:ul22v+Nro/R083muKhosV54bj5niojjWZvU8xrevuH4= +github.com/yuin/goldmark v1.1.25/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= +github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= +github.com/yuin/goldmark v1.1.32/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= +github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY= +github.com/yunify/qingstor-sdk-go/v3 v3.2.0 h1:9sB2WZMgjwSUNZhrgvaNGazVltoFUUfuS9f0uCWtTr8= +github.com/yunify/qingstor-sdk-go/v3 v3.2.0/go.mod h1:KciFNuMu6F4WLk9nGwwK69sCGKLCdd9f97ac/wfumS4= github.com/yusufpapurcu/wmi v1.2.2 h1:KBNDSne4vP5mbSWnJbO+51IMOXJB67QiYCSBrubbPRg= github.com/yusufpapurcu/wmi v1.2.2/go.mod h1:SBZ9tNy3G9/m5Oi98Zks0QjeHVDvuK0qfxQmPyzfmi0= +github.com/zeebo/assert v1.3.0 h1:g7C04CbJuIDKNPFHmsk4hwZDO5O+kntRxzaUoNXj+IQ= +github.com/zeebo/errs v1.3.0 h1:hmiaKqgYZzcVgRL1Vkc1Mn2914BbzB0IBxs+ebeutGs= +github.com/zeebo/errs v1.3.0/go.mod h1:sgbWHsvVuTPHcqJJGQ1WhI5KbWlHYz+2+2C/LSEtCw4= +go.etcd.io/bbolt v1.3.6 h1:/ecaJf0sk1l4l6V4awd65v2C3ILy7MSj+s/x1ADCIMU= +go.etcd.io/bbolt v1.3.6/go.mod h1:qXsaaIqmgQH0T+OPdb99Bf+PKfBBQVAdyD6TY9G8XM4= +go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU= +go.opencensus.io v0.22.0/go.mod h1:+kGneAE2xo2IficOXnaByMWTGM9T73dGwxeWcUqIpI8= +go.opencensus.io v0.22.2/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= +go.opencensus.io v0.22.3/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= +go.opencensus.io v0.22.4/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= +go.opencensus.io v0.24.0 h1:y73uSU6J157QMP2kn2r30vwW1A2W2WFwSCGnAVxeaD0= +go.opencensus.io v0.24.0/go.mod h1:vNK8G9p7aAivkbmorf4v+7Hgx+Zs0yY+0fOtgBfjQKo= go.uber.org/atomic v1.7.0 h1:ADUqmZGgLDDfbSL9ZmPxKTybcoEYHgpYfELNoN+7hsw= go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc= go.uber.org/goleak v1.1.11 h1:wy28qYRKZgnJTxGxvye5/wgWr1EKjmUDGYox5mGlRlI= @@ -290,107 +658,375 @@ go.uber.org/multierr v1.6.0 h1:y6IPFStTAIT5Ytl7/XYmHvzXQ7S3g/IeZW9hyZ5thw4= go.uber.org/multierr v1.6.0/go.mod h1:cdWPpRnG4AhwMwsgIHip0KRBQjJy5kYEpYjJxpXp9iU= go.uber.org/zap v1.24.0 h1:FiJd5l1UOLj0wCgbSE0rwwXHzEdAZS6hiiSnxJN/D60= go.uber.org/zap v1.24.0/go.mod h1:2kMP+WWQ8aoFoedH3T2sq6iJ2yDWpHbP0f6MQbS9Gkg= +golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= +golang.org/x/crypto v0.0.0-20190131182504-b8fe1690c613/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= +golang.org/x/crypto v0.0.0-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/crypto v0.0.0-20190605123033-f99c8df09eb5/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/crypto v0.0.0-20200302210943-78000ba7a073/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= golang.org/x/crypto v0.0.0-20200728195943-123391ffb6de/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= golang.org/x/crypto v0.0.0-20210711020723-a769d52b0f97/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= +golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= golang.org/x/crypto v0.0.0-20211215153901-e495a2d5b3d3/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= +golang.org/x/crypto v0.0.0-20220622213112-05595931fe9d/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= +golang.org/x/crypto v0.0.0-20220722155217-630584e8d5aa/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= golang.org/x/crypto v0.5.0 h1:U/0M97KRkSFvyD/3FSmdP5W5swImpNgle/EHFhOsQPE= golang.org/x/crypto v0.5.0/go.mod h1:NK/OQwhpMQP3MwtdjgLlYHnH9ebylxKWv3e0fK+mkQU= +golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= +golang.org/x/exp v0.0.0-20190306152737-a1d7652674e8/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= +golang.org/x/exp v0.0.0-20190510132918-efd6b22b2522/go.mod h1:ZjyILWgesfNpC6sMxTJOJm9Kp84zZh5NQWvqDGG3Qr8= +golang.org/x/exp v0.0.0-20190829153037-c13cbed26979/go.mod h1:86+5VVa7VpoJ4kLfm080zCjGlMRFzhUhsZKEZO7MGek= +golang.org/x/exp v0.0.0-20191030013958-a1ab85dbe136/go.mod h1:JXzH8nQsPlswgeRAPE3MuO9GYsAcnJvJ4vnMwN/5qkY= +golang.org/x/exp v0.0.0-20191129062945-2f5052295587/go.mod h1:2RIsYlXP63K8oxa1u096TMicItID8zy7Y6sNkU49FU4= +golang.org/x/exp v0.0.0-20191227195350-da58074b4299/go.mod h1:2RIsYlXP63K8oxa1u096TMicItID8zy7Y6sNkU49FU4= +golang.org/x/exp v0.0.0-20200119233911-0405dc783f0a/go.mod h1:2RIsYlXP63K8oxa1u096TMicItID8zy7Y6sNkU49FU4= +golang.org/x/exp v0.0.0-20200207192155-f17229e696bd/go.mod h1:J/WKrq2StrnmMY6+EHIKF9dgMWnmCNThgcyBT1FY9mM= +golang.org/x/exp v0.0.0-20200224162631-6cc2880d07d6/go.mod h1:3jZMyOhIsHpP37uCMkUooju7aAi5cS1Q23tOzKc+0MU= +golang.org/x/image v0.0.0-20190227222117-0694c2d4d067/go.mod h1:kZ7UVZpmo3dzQBMxlp+ypCbDeSB+sBbTgSJuh5dn5js= +golang.org/x/image v0.0.0-20190802002840-cff245a6509b/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= golang.org/x/image v0.0.0-20191009234506-e7c1f5e7dbb8 h1:hVwzHzIUGRjiF7EcUjqNxk3NCfkPxbDKRdnNE1Rpg0U= golang.org/x/image v0.0.0-20191009234506-e7c1f5e7dbb8/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= +golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= +golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU= +golang.org/x/lint v0.0.0-20190301231843-5614ed5bae6f/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= +golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= +golang.org/x/lint v0.0.0-20190409202823-959b441ac422/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= +golang.org/x/lint v0.0.0-20190909230951-414d861bb4ac/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= +golang.org/x/lint v0.0.0-20190930215403-16217165b5de/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= +golang.org/x/lint v0.0.0-20191125180803-fdd1cda4f05f/go.mod h1:5qLYkcX4OjUUV8bRuDixDT3tpyyb+LUpUlRWLxfhWrs= +golang.org/x/lint v0.0.0-20200130185559-910be7a94367/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY= +golang.org/x/lint v0.0.0-20200302205851-738671d3881b/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY= +golang.org/x/mobile v0.0.0-20190312151609-d3739f865fa6/go.mod h1:z+o9i4GpDbdi3rU15maQ/Ox0txvL9dWGYEHz965HBQE= +golang.org/x/mobile v0.0.0-20190719004257-d2bd2a29d028/go.mod h1:E/iHnbuqvinMTCcRqshq8CkpyQDoeVncDDYHnLhea+o= +golang.org/x/mod v0.0.0-20190513183733-4bf6d317e70e/go.mod h1:mXi4GBBbnImb6dmsKGUJ2LatrhH/nqhxcFungHvyanc= +golang.org/x/mod v0.1.0/go.mod h1:0QHyrYULN0/3qlju5TqG8bIK38QM8yzMo5ekMj3DlcY= +golang.org/x/mod v0.1.1-0.20191105210325-c90efee705ee/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg= +golang.org/x/mod v0.1.1-0.20191107180719-034126e5016b/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg= +golang.org/x/mod v0.2.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= +golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= +golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20181114220301-adae6a3d119a/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= +golang.org/x/net v0.0.0-20190501004415-9ce7a6920f09/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= +golang.org/x/net v0.0.0-20190503192946-f4e77d36d62c/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= golang.org/x/net v0.0.0-20190603091049-60506f45cf65/go.mod h1:HSz+uSET+XFnRR8LxR5pz3Of3rY3CfYBVs4xY44aLks= +golang.org/x/net v0.0.0-20190613194153-d28f0bde5980/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20190628185345-da137c7871d7/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20190724013045-ca1201d0de80/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20190923162816-aa69164e4478/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20191209160850-c0dbc17a3553/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20200114155413-6afb5195e5aa/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20200202094626-16171245cfb2/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20200222125558-5a598a2470a0/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20200226121028-0de0cce0169b/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20200301022130-244492dfa37a/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20200320220750-118fecf932d8/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20200324143707-d3edc9973b7e/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= golang.org/x/net v0.0.0-20200501053045-e0ff5e5a1de5/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= +golang.org/x/net v0.0.0-20200506145744-7e3656a0809f/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= golang.org/x/net v0.0.0-20200513185701-a91f0712d120/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= +golang.org/x/net v0.0.0-20200520182314-0ba52f642ac2/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= +golang.org/x/net v0.0.0-20200625001655-4c5254603344/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= +golang.org/x/net v0.0.0-20200707034311-ab3426394381/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= +golang.org/x/net v0.0.0-20200822124328-c89045814202/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= +golang.org/x/net v0.0.0-20201110031124-69a78807bb2b/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= +golang.org/x/net v0.0.0-20210525063256-abc453219eb5/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= golang.org/x/net v0.0.0-20211029224645-99673261e6eb/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= +golang.org/x/net v0.0.0-20220127200216-cd36cc0744dd/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk= +golang.org/x/net v0.0.0-20220225172249-27dd8689420f/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk= +golang.org/x/net v0.0.0-20220624214902-1bab6f366d9e/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c= +golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c= +golang.org/x/net v0.0.0-20220725212005-46097bf591d3/go.mod h1:AaygXjzTFtRAg2ttMY5RMuhpJ3cNnI0XpyFJD1iQRSM= golang.org/x/net v0.0.0-20221002022538-bcab6841153b/go.mod h1:YDH+HFinaLZZlnHAfSS6ZXJJ9M9t4Dl22yv3iI2vPwk= +golang.org/x/net v0.1.0/go.mod h1:Cx3nUiGt4eDBEyega/BKRp+/AlGL8hYe7U9odMt2Cco= golang.org/x/net v0.5.0 h1:GyT4nK/YDHSqa1c4753ouYCDajOYKTja9Xb/OHtgvSw= golang.org/x/net v0.5.0/go.mod h1:DivGGAXEgPSlEBzxGzZI+ZLohi+xUj054jfeKui00ws= golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= +golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= +golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= +golang.org/x/oauth2 v0.0.0-20191202225959-858c2ad4c8b6/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= +golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= +golang.org/x/oauth2 v0.0.0-20201208152858-08078c50e5b5/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= +golang.org/x/oauth2 v0.0.0-20210514164344-f6687ab2804c/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= +golang.org/x/oauth2 v0.0.0-20220223155221-ee480838109b/go.mod h1:DAh4E804XQdzx2j+YRIaUnCqCV2RuMz24cGBJ5QYIrc= golang.org/x/oauth2 v0.3.0 h1:6l90koy8/LaBLmLu8jpHeHexzMwEita0zFfYlggy2F8= golang.org/x/oauth2 v0.3.0/go.mod h1:rQrIauxkUhJ6CuwEXwymO2/eh4xz2ZWF1nBkcxS+tGk= +golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20190227155943-e225da77a7e6/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20200317015054-43a5402ce75a/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20200625203802-6e8e738ad208/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sync v0.0.0-20210220032951-036812b2e83c h1:5KslGYwFpkhGh+Q16bwMP3cOontH8FOep7tGV86Y7SQ= +golang.org/x/sync v0.0.0-20201207232520-09787c993a3a/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.1.0 h1:wsuoTGHzEhffawBOhz5CYhcrV4IdKZbEyZjBMuTp12o= +golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20181116152217-5ac8a444bdc5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190502145724-3ef323f4f1fd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190507160741-ecd444e8653b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190606165138-5da285871e9c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190624142023-c5567b49c5d0/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190726091711-fc99dfbffb4e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190916202348-b4ddaad3f8a3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191001151750-bb3f8db39f24/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191120155948-bd437916bb0e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191210023423-ac6580df4449/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191228213918-04cbcbbfeed8/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200106162015-b016eb3dc98e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200113162924-86b910548bc1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200122134326-e047566fdf82/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200202164722-d101bd2416d5/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200212091648-12a6c2dcc1e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200302150141-5c8b2ff67527/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200331124033-c3d80250170d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200501052902-10377860bb8e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200511232937-7e40ca221e25/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200515095857-1151b9dac4a9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200523222454-059865788121/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200615200032-f1bc736245b1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200625212154-ddb9806d33ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200803210538-64077c9b5642/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200923182605-d9f96fdee20d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20201204225414-ed752295db88/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210124154548-22da62e12c0c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210303074136-134d130e1a04/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210320140829-1e4c9ba3b0c4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20210514084401-e8d321eab015/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20210603081109-ebe580a85c40/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210806184541-e5e7981a1069/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210927094055-39ccf1dd6fa6/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20211103235746-7861aae1554b/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20220114195835-da31bd327af9/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20220209214540-3681064d5158/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220310020820-b874c991c1a5/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20220408201424-a24fb2fb8a0f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220728004956-3c1f35247d10/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220928140112-f11e5e49a4ec/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.1.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.2.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.4.0 h1:Zr2JFtRQNX3BCZ8YtxRE9hNJYC8J6I1MVbMg6owUp18= golang.org/x/sys v0.4.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= +golang.org/x/term v0.1.0/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= golang.org/x/term v0.4.0 h1:O7UWfv5+A2qiuulQk30kVinPoMtoIPeVaKLEgLpVkvg= +golang.org/x/term v0.4.0/go.mod h1:9P2UbLfCdcvo3p/nzKvsmas4TnlujnuoV9hGgYzW1lQ= +golang.org/x/text v0.0.0-20170915032832-14c0d48ead0c/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= +golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= +golang.org/x/text v0.4.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= golang.org/x/text v0.6.0 h1:3XmdazWV+ubf7QgHSTWeykHOci5oeekaGJBLkrkaw4k= golang.org/x/text v0.6.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= +golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= +golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= +golang.org/x/time v0.0.0-20191024005414-555d28b269f0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= golang.org/x/time v0.2.0 h1:52I/1L54xyEQAYdtcSuxtiT84KGYTBGXwayxmIpNJhE= golang.org/x/time v0.2.0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= +golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= +golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY= +golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= +golang.org/x/tools v0.0.0-20190312151545-0bb0c0a6e846/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= +golang.org/x/tools v0.0.0-20190312170243-e65039ee4138/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= +golang.org/x/tools v0.0.0-20190328211700-ab21143f2384/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= +golang.org/x/tools v0.0.0-20190425150028-36563e24a262/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= +golang.org/x/tools v0.0.0-20190506145303-2d16b83fe98c/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= +golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= +golang.org/x/tools v0.0.0-20190606124116-d0a3d012864b/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= +golang.org/x/tools v0.0.0-20190621195816-6e04913cbbac/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= +golang.org/x/tools v0.0.0-20190628153133-6cdbf07be9d0/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= +golang.org/x/tools v0.0.0-20190816200558-6889da9d5479/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= +golang.org/x/tools v0.0.0-20190911174233-4f2ddba30aff/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= +golang.org/x/tools v0.0.0-20191012152004-8de300cfc20a/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= +golang.org/x/tools v0.0.0-20191113191852-77e3bb0ad9e7/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= +golang.org/x/tools v0.0.0-20191115202509-3a792d9c32b2/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= +golang.org/x/tools v0.0.0-20191125144606-a911d9008d1f/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= +golang.org/x/tools v0.0.0-20191130070609-6e064ea0cf2d/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= +golang.org/x/tools v0.0.0-20191216173652-a0e659d51361/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= +golang.org/x/tools v0.0.0-20191227053925-7b8e75db28f4/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= +golang.org/x/tools v0.0.0-20200117161641-43d50277825c/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= +golang.org/x/tools v0.0.0-20200122220014-bf1340f18c4a/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= +golang.org/x/tools v0.0.0-20200130002326-2f3ba24bd6e7/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= +golang.org/x/tools v0.0.0-20200204074204-1cc6d1ef6c74/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= +golang.org/x/tools v0.0.0-20200207183749-b753a1ba74fa/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= +golang.org/x/tools v0.0.0-20200212150539-ea181f53ac56/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= +golang.org/x/tools v0.0.0-20200224181240-023911ca70b2/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= +golang.org/x/tools v0.0.0-20200227222343-706bc42d1f0d/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= +golang.org/x/tools v0.0.0-20200304193943-95d2e580d8eb/go.mod h1:o4KQGtdN14AW+yjsvvwRTJJuXz8XRtIHtEnmAXLyFUw= +golang.org/x/tools v0.0.0-20200312045724-11d5b4c81c7d/go.mod h1:o4KQGtdN14AW+yjsvvwRTJJuXz8XRtIHtEnmAXLyFUw= +golang.org/x/tools v0.0.0-20200331025713-a30bf2db82d4/go.mod h1:Sl4aGygMT6LrqrWclx+PTx3U+LnKx/seiNR+3G19Ar8= +golang.org/x/tools v0.0.0-20200423201157-2723c5de0d66/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= +golang.org/x/tools v0.0.0-20200501065659-ab2804fb9c9d/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= +golang.org/x/tools v0.0.0-20200512131952-2bc93b1c0c88/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= +golang.org/x/tools v0.0.0-20200515010526-7d3b6ebf133d/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= +golang.org/x/tools v0.0.0-20200618134242-20370b0cb4b2/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= +golang.org/x/tools v0.0.0-20200619180055-7c47624df98f/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= +golang.org/x/tools v0.0.0-20200729194436-6467de6f59a7/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= +golang.org/x/tools v0.0.0-20200804011535-6c149bb5ef0d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= +golang.org/x/tools v0.0.0-20200825202427-b303f430e36d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= golang.org/x/tools v0.0.0-20201124115921-2c860bdd6e78/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= +golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.1.1/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= +golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +golang.org/x/xerrors v0.0.0-20220907171357-04be3eba64a2 h1:H2TDz8ibqkAF6YGhCdN3jS9O0/s90v0rJh3X/OLHEUk= +google.golang.org/api v0.4.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE= +google.golang.org/api v0.7.0/go.mod h1:WtwebWUNSVBH/HAw79HIFXZNqEvBhG+Ra+ax0hx3E3M= +google.golang.org/api v0.8.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg= +google.golang.org/api v0.9.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg= +google.golang.org/api v0.13.0/go.mod h1:iLdEw5Ide6rF15KTC1Kkl0iskquN2gFfn9o9XIsbkAI= +google.golang.org/api v0.14.0/go.mod h1:iLdEw5Ide6rF15KTC1Kkl0iskquN2gFfn9o9XIsbkAI= +google.golang.org/api v0.15.0/go.mod h1:iLdEw5Ide6rF15KTC1Kkl0iskquN2gFfn9o9XIsbkAI= +google.golang.org/api v0.17.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE= +google.golang.org/api v0.18.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE= +google.golang.org/api v0.19.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE= +google.golang.org/api v0.20.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE= +google.golang.org/api v0.22.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE= +google.golang.org/api v0.24.0/go.mod h1:lIXQywCXRcnZPGlsd8NbLnOjtAoL6em04bJ9+z0MncE= +google.golang.org/api v0.28.0/go.mod h1:lIXQywCXRcnZPGlsd8NbLnOjtAoL6em04bJ9+z0MncE= +google.golang.org/api v0.29.0/go.mod h1:Lcubydp8VUV7KeIHD9z2Bys/sm/vGKnG1UHuDBSrHWM= +google.golang.org/api v0.30.0/go.mod h1:QGmEvQ87FHZNiUVJkT14jQNYJ4ZJjdRF23ZXz5138Fc= +google.golang.org/api v0.103.0 h1:9yuVqlu2JCvcLg9p8S3fcFLZij8EPSyvODIY1rkMizQ= +google.golang.org/api v0.103.0/go.mod h1:hGtW6nK1AC+d9si/UBhw8Xli+QMOf6xyNAyJw4qU9w0= google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= +google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= +google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= +google.golang.org/appengine v1.6.1/go.mod h1:i06prIuMbXzDqacNJfV5OdTW448YApPu5ww/cMBSeb0= +google.golang.org/appengine v1.6.5/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= +google.golang.org/appengine v1.6.6/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= google.golang.org/appengine v1.6.7 h1:FZR1q0exgwxzPzp/aF+VccGrSfxfPpkBqjIIEq3ru6c= google.golang.org/appengine v1.6.7/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= +google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= +google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= +google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= +google.golang.org/genproto v0.0.0-20190425155659-357c62f0e4bb/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= +google.golang.org/genproto v0.0.0-20190502173448-54afdca5d873/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= +google.golang.org/genproto v0.0.0-20190801165951-fa694d86fc64/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= +google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= +google.golang.org/genproto v0.0.0-20190911173649-1774047e7e51/go.mod h1:IbNlFCBrqXvoKpeg0TB2l7cyZUmoaFKYIwrEpbDKLA8= +google.golang.org/genproto v0.0.0-20191108220845-16a3f7862a1a/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= +google.golang.org/genproto v0.0.0-20191115194625-c23dd37a84c9/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= +google.golang.org/genproto v0.0.0-20191216164720-4f79533eabd1/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= +google.golang.org/genproto v0.0.0-20191230161307-f3c370f40bfb/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= +google.golang.org/genproto v0.0.0-20200115191322-ca5a22157cba/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= +google.golang.org/genproto v0.0.0-20200122232147-0452cf42e150/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= +google.golang.org/genproto v0.0.0-20200204135345-fa8e72b47b90/go.mod h1:GmwEX6Z4W5gMy59cAlVYjN9JhxgbQH6Gn+gFDQe2lzA= +google.golang.org/genproto v0.0.0-20200212174721-66ed5ce911ce/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= +google.golang.org/genproto v0.0.0-20200224152610-e50cd9704f63/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= +google.golang.org/genproto v0.0.0-20200228133532-8c2c7df3a383/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= +google.golang.org/genproto v0.0.0-20200305110556-506484158171/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= +google.golang.org/genproto v0.0.0-20200312145019-da6875a35672/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= +google.golang.org/genproto v0.0.0-20200331122359-1ee6d9798940/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= +google.golang.org/genproto v0.0.0-20200430143042-b979b6f78d84/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= +google.golang.org/genproto v0.0.0-20200511104702-f5ebc3bea380/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= +google.golang.org/genproto v0.0.0-20200515170657-fc4c6c6a6587/go.mod h1:YsZOwe1myG/8QRHRsmBRE1LrgQY60beZKjly0O1fX9U= +google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo= +google.golang.org/genproto v0.0.0-20200618031413-b414f8b61790/go.mod h1:jDfRM7FcilCzHH/e9qn6dsT145K34l5v+OpcnNgKAAA= +google.golang.org/genproto v0.0.0-20200729003335-053ba62fc06f/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20200804131852-c06518451d9c/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20200825200019-8632dd797987/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20221027153422-115e99e71e1c h1:QgY/XxIAIeccR+Ca/rDdKubLIU9rcJ3xfy1DC/Wd2Oo= +google.golang.org/genproto v0.0.0-20221027153422-115e99e71e1c/go.mod h1:CGI5F/G+E5bKwmfYo09AXuVN4dD894kIKUFmVbP2/Fo= +google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c= +google.golang.org/grpc v1.20.1/go.mod h1:10oTOabMzJvdu6/UiuZezV6QK5dSlG84ov/aaiqXj38= +google.golang.org/grpc v1.21.1/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM= +google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= +google.golang.org/grpc v1.25.1/go.mod h1:c3i+UQWmh7LiEpx4sFZnkU36qjEYZ0imhYfXVyQciAY= +google.golang.org/grpc v1.26.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= +google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= +google.golang.org/grpc v1.27.1/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= +google.golang.org/grpc v1.28.0/go.mod h1:rpkK4SK4GF4Ach/+MFLZUBavHOvF2JJB5uozKKal+60= +google.golang.org/grpc v1.29.1/go.mod h1:itym6AZVZYACWQqET3MqgPpjcuV5QH3BxFS3IjizoKk= +google.golang.org/grpc v1.30.0/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= +google.golang.org/grpc v1.31.0/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= +google.golang.org/grpc v1.33.2/go.mod h1:JMHMWHQWaTccqQQlmk3MJZS+GWXOdAesneDmEnv2fbc= +google.golang.org/grpc v1.50.1 h1:DS/BukOZWp8s6p4Dt/tOaJaTQyPyOoCcrjroHuCeLzY= +google.golang.org/grpc v1.50.1/go.mod h1:ZgQEeidpAuNRZ8iRrlBKXZQP1ghovWIVhdJRyCDK+GI= +google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8= +google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= +google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM= +google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE= +google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo= +google.golang.org/protobuf v1.22.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= +google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= +google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= +google.golang.org/protobuf v1.24.0/go.mod h1:r/3tXBNzIEhYS9I1OUVjXDlt8tc493IdKGjtUeSXeh4= +google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c= google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= google.golang.org/protobuf v1.28.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I= google.golang.org/protobuf v1.28.1 h1:d0NfwRgPtno5B1Wa6L2DAG+KivqkdutMf1UhdNx175w= google.golang.org/protobuf v1.28.1/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I= +gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= +gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= gopkg.in/ini.v1 v1.67.0 h1:Dgnx+6+nfE+IfzjUEISNeydPJh9AXNNsWbGP9KzCsOA= gopkg.in/ini.v1 v1.67.0/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k= gopkg.in/natefinch/lumberjack.v2 v2.0.0 h1:1Lc07Kr7qY4U2YPouBjpCLxpiyxIVoxqXgkXLknAOE8= gopkg.in/natefinch/lumberjack.v2 v2.0.0/go.mod h1:l0ndWWf7gzL7RNwBG7wST/UCcT4T24xpD6X8LsfU/+k= +gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ= +gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= +gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.2.5/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.7/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= @@ -406,6 +1042,13 @@ gorm.io/gorm v1.24.3/go.mod h1:DVrVomtaYTbqs7gB/x2uVvqnXzv0nqjB396B8cG4dBA= gotest.tools v2.2.0+incompatible h1:VsBPFP1AI068pPrMxtb/S8Zkgf9xEmTLJjfM+P5UIEo= gotest.tools v2.2.0+incompatible/go.mod h1:DsYFclhRJ6vuDpmuTbkuFWG+y2sxOXAzmJt81HFBacw= gotest.tools/v3 v3.3.0 h1:MfDY1b1/0xN1CyMlQDac0ziEy9zJQd9CXBRRDHw2jJo= +honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= +honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= +honnef.co/go/tools v0.0.0-20190418001031-e561f6794a2a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= +honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= +honnef.co/go/tools v0.0.1-2019.2.3/go.mod h1:a3bituU0lyd329TUQxRnasdCoJDkEUEAqEt0JzvZhAg= +honnef.co/go/tools v0.0.1-2020.1.3/go.mod h1:X/FiERA/W4tHapMX5mGpAtMSVEeEUOyHaw9vFzvIQ3k= +honnef.co/go/tools v0.0.1-2020.1.4/go.mod h1:X/FiERA/W4tHapMX5mGpAtMSVEeEUOyHaw9vFzvIQ3k= lukechampine.com/uint128 v1.1.1/go.mod h1:c4eWIwlEGaxC/+H1VguhU4PHXNWDCDMUlWdIWl2j1gk= lukechampine.com/uint128 v1.2.0/go.mod h1:c4eWIwlEGaxC/+H1VguhU4PHXNWDCDMUlWdIWl2j1gk= modernc.org/cc/v3 v3.37.0/go.mod h1:vtL+3mdHx/wcj3iEGz84rQa8vEqR6XM84v5Lcvfph20= @@ -437,3 +1080,12 @@ modernc.org/strutil v1.1.3/go.mod h1:MEHNA7PdEnEwLvspRMtWTNnp2nnyvMfkimT1NKNAGbw modernc.org/tcl v1.15.0/go.mod h1:xRoGotBZ6dU+Zo2tca+2EqVEeMmOUBzHnhIwq4YrVnE= modernc.org/token v1.0.1/go.mod h1:UGzOrNV1mAFSEB63lOFHIpNRUVMvYTc6yu1SMY/XTDM= modernc.org/z v1.7.0/go.mod h1:hVdgNMh8ggTuRG1rGU8x+xGRFfiQUIAw0ZqlPy8+HyQ= +rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8= +rsc.io/quote/v3 v3.1.0/go.mod h1:yEA65RcK8LyAZtP9Kv3t0HmxON59tX3rD+tICJqUlj0= +rsc.io/sampler v1.3.0/go.mod h1:T1hPZKmBbMNahiBKFy5HrXp6adAjACjK9JXDnKaTXpA= +storj.io/common v0.0.0-20220414110316-a5cb7172d6bf h1:D5xZTDOlTTQWdAWeKKm2pFLcz1sceH+f/pVAcYB9jL8= +storj.io/common v0.0.0-20220414110316-a5cb7172d6bf/go.mod h1:LBJrpAqL4MNSrhGEwc8SJ+tIVtgfCtFEZqDy6/0j67A= +storj.io/drpc v0.0.30 h1:jqPe4T9KEu3CDBI05A2hCMgMSHLtd/E0N0yTF9QreIE= +storj.io/drpc v0.0.30/go.mod h1:6rcOyR/QQkSTX/9L5ZGtlZaE2PtXTTZl8d+ulSeeYEg= +storj.io/uplink v1.9.0 h1:Zg1kX1VqOQIKm0yAukteKpLuT68Be3euyNRML612ERM= +storj.io/uplink v1.9.0/go.mod h1:f6D8306j5mnRHnPDKWCiwtPM6ukyGg77to9LaAY9l6k= diff --git a/internal/conf/config.go b/internal/conf/config.go deleted file mode 100644 index 16dabae..0000000 --- a/internal/conf/config.go +++ /dev/null @@ -1,43 +0,0 @@ -package conf - -type Database struct { - Type string `json:"type" env:"DB_TYPE"` - Host string `json:"host" env:"DB_HOST"` - Port int `json:"port" env:"DB_PORT"` - User string `json:"user" env:"DB_USER"` - Password string `json:"password" env:"DB_PASS"` - Name string `json:"name" env:"DB_NAME"` - DBFile string `json:"db_file" env:"DB_FILE"` - TablePrefix string `json:"table_prefix" env:"DB_TABLE_PREFIX"` - SSLMode string `json:"ssl_mode" env:"DB_SSL_MODE"` -} - -type Scheme struct { - Https bool `json:"https" env:"HTTPS"` - CertFile string `json:"cert_file" env:"CERT_FILE"` - KeyFile string `json:"key_file" env:"KEY_FILE"` -} - -type LogConfig struct { - Enable bool `json:"enable" env:"LOG_ENABLE"` - Name string `json:"name" env:"LOG_NAME"` - MaxSize int `json:"max_size" env:"MAX_SIZE"` - MaxBackups int `json:"max_backups" env:"MAX_BACKUPS"` - MaxAge int `json:"max_age" env:"MAX_AGE"` - Compress bool `json:"compress" env:"COMPRESS"` -} - -type Config struct { - Force bool `json:"force" env:"FORCE"` - Address string `json:"address" env:"ADDR"` - Port int `json:"port" env:"PORT"` - SiteURL string `json:"site_url" env:"SITE_URL"` - Cdn string `json:"cdn" env:"CDN"` - JwtSecret string `json:"jwt_secret" env:"JWT_SECRET"` - TokenExpiresIn int `json:"token_expires_in" env:"TOKEN_EXPIRES_IN"` - Database Database `json:"database"` - Scheme Scheme `json:"scheme"` - TempDir string `json:"temp_dir" env:"TEMP_DIR"` - BleveDir string `json:"bleve_dir" env:"BLEVE_DIR"` - Log LogConfig `json:"log"` -} diff --git a/internal/conf/const.go b/internal/conf/const.go deleted file mode 100644 index 6a75634..0000000 --- a/internal/conf/const.go +++ /dev/null @@ -1,72 +0,0 @@ -package conf - -const ( - TypeString = "string" - TypeSelect = "select" - TypeBool = "bool" - TypeText = "text" - TypeNumber = "number" -) - -const ( - // site - VERSION = "version" - ApiUrl = "api_url" - BasePath = "base_path" - SiteTitle = "site_title" - Announcement = "announcement" - AllowIndexed = "allow_indexed" - - Logo = "logo" - Favicon = "favicon" - MainColor = "main_color" - - // preview - TextTypes = "text_types" - AudioTypes = "audio_types" - VideoTypes = "video_types" - ImageTypes = "image_types" - ProxyTypes = "proxy_types" - ProxyIgnoreHeaders = "proxy_ignore_headers" - AudioAutoplay = "audio_autoplay" - VideoAutoplay = "video_autoplay" - - // global - HideFiles = "hide_files" - CustomizeHead = "customize_head" - CustomizeBody = "customize_body" - LinkExpiration = "link_expiration" - SignAll = "sign_all" - PrivacyRegs = "privacy_regs" - OcrApi = "ocr_api" - FilenameCharMapping = "filename_char_mapping" - - // index - SearchIndex = "search_index" - AutoUpdateIndex = "auto_update_index" - IndexPaths = "index_paths" - IgnorePaths = "ignore_paths" - - // aria2 - Aria2Uri = "aria2_uri" - Aria2Secret = "aria2_secret" - - // single - Token = "token" - IndexProgress = "index_progress" - - //Github - GithubClientId = "github_client_id" - GithubClientSecrets = "github_client_secrets" - GithubLoginEnabled = "github_login_enabled" -) - -const ( - UNKNOWN = iota - FOLDER - //OFFICE - VIDEO - AUDIO - TEXT - IMAGE -) diff --git a/internal/conf/var.go b/internal/conf/var.go deleted file mode 100644 index a91547d..0000000 --- a/internal/conf/var.go +++ /dev/null @@ -1,30 +0,0 @@ -package conf - -import "regexp" - -var ( - BuiltAt string - GoVersion string - GitAuthor string - GitCommit string - Version string = "dev" - WebVersion string -) - -var ( - Conf *Config -) - -var SlicesMap = make(map[string][]string) -var FilenameCharMap = make(map[string]string) -var PrivacyReg []*regexp.Regexp - -var ( - // StoragesLoaded loaded success if empty - StoragesLoaded = false -) -var ( - RawIndexHtml string - ManageHtml string - IndexHtml string -) diff --git a/internal/driver/config.go b/internal/driver/config.go deleted file mode 100644 index a7758ba..0000000 --- a/internal/driver/config.go +++ /dev/null @@ -1,25 +0,0 @@ -/* - * @Author: a624669980@163.com a624669980@163.com - * @Date: 2022-12-13 11:05:05 - * @LastEditors: a624669980@163.com a624669980@163.com - * @LastEditTime: 2022-12-13 11:05:13 - * @FilePath: /drive/internal/driver/config.go - * @Description: 这是默认设置,请设置`customMade`, 打开koroFileHeader查看配置 进行设置: https://github.com/OBKoro1/koro1FileHeader/wiki/%E9%85%8D%E7%BD%AE - */ -package driver - -type Config struct { - Name string `json:"name"` - LocalSort bool `json:"local_sort"` - OnlyLocal bool `json:"only_local"` - OnlyProxy bool `json:"only_proxy"` - NoCache bool `json:"no_cache"` - NoUpload bool `json:"no_upload"` - NeedMs bool `json:"need_ms"` // if need get message from user, such as validate code - DefaultRoot string `json:"default_root"` - CheckStatus bool -} - -func (c Config) MustProxy() bool { - return c.OnlyProxy || c.OnlyLocal -} diff --git a/internal/driver/driver.go b/internal/driver/driver.go deleted file mode 100644 index b33bf80..0000000 --- a/internal/driver/driver.go +++ /dev/null @@ -1,131 +0,0 @@ -package driver - -import ( - "context" - - "github.com/IceWhaleTech/CasaOS/model" -) - -type Driver interface { - Meta - Reader - User - //Writer - //Other -} - -type Meta interface { - Config() Config - // GetStorage just get raw storage, no need to implement, because model.Storage have implemented - GetStorage() *model.Storage - SetStorage(model.Storage) - // GetAddition Additional is used for unmarshal of JSON, so need return pointer - GetAddition() Additional - // Init If already initialized, drop first - Init(ctx context.Context) error - Drop(ctx context.Context) error -} - -type Other interface { - Other(ctx context.Context, args model.OtherArgs) (interface{}, error) -} - -type Reader interface { - // List files in the path - // if identify files by path, need to set ID with path,like path.Join(dir.GetID(), obj.GetName()) - // if identify files by id, need to set ID with corresponding id - List(ctx context.Context, dir model.Obj, args model.ListArgs) ([]model.Obj, error) - // Link get url/filepath/reader of file - Link(ctx context.Context, file model.Obj, args model.LinkArgs) (*model.Link, error) -} -type User interface { - // GetRoot get root directory of user - GetUserInfo(ctx context.Context) (string, error) -} -type Getter interface { - GetRoot(ctx context.Context) (model.Obj, error) -} - -//type Writer interface { -// Mkdir -// Move -// Rename -// Copy -// Remove -// Put -//} - -type Mkdir interface { - MakeDir(ctx context.Context, parentDir model.Obj, dirName string) error -} - -type Move interface { - Move(ctx context.Context, srcObj, dstDir model.Obj) error -} - -type Rename interface { - Rename(ctx context.Context, srcObj model.Obj, newName string) error -} - -type Copy interface { - Copy(ctx context.Context, srcObj, dstDir model.Obj) error -} - -type Remove interface { - Remove(ctx context.Context, obj model.Obj) error -} - -type Put interface { - Put(ctx context.Context, dstDir model.Obj, stream model.FileStreamer, up UpdateProgress) error -} - -//type WriteResult interface { -// MkdirResult -// MoveResult -// RenameResult -// CopyResult -// PutResult -// Remove -//} - -type MkdirResult interface { - MakeDir(ctx context.Context, parentDir model.Obj, dirName string) (model.Obj, error) -} - -type MoveResult interface { - Move(ctx context.Context, srcObj, dstDir model.Obj) (model.Obj, error) -} - -type RenameResult interface { - Rename(ctx context.Context, srcObj model.Obj, newName string) (model.Obj, error) -} - -type CopyResult interface { - Copy(ctx context.Context, srcObj, dstDir model.Obj) (model.Obj, error) -} - -type PutResult interface { - Put(ctx context.Context, dstDir model.Obj, stream model.FileStreamer, up UpdateProgress) (model.Obj, error) -} - -type UpdateProgress func(percentage int) - -type Progress struct { - Total int64 - Done int64 - up UpdateProgress -} - -func (p *Progress) Write(b []byte) (n int, err error) { - n = len(b) - p.Done += int64(n) - p.up(int(float64(p.Done) / float64(p.Total) * 100)) - return -} - -func NewProgress(total int64, up UpdateProgress) *Progress { - return &Progress{ - Total: total, - up: up, - } -} diff --git a/internal/driver/item.go b/internal/driver/item.go deleted file mode 100644 index 6dff49d..0000000 --- a/internal/driver/item.go +++ /dev/null @@ -1,56 +0,0 @@ -/* - * @Author: a624669980@163.com a624669980@163.com - * @Date: 2022-12-13 11:05:47 - * @LastEditors: a624669980@163.com a624669980@163.com - * @LastEditTime: 2022-12-13 11:05:54 - * @FilePath: /drive/internal/driver/item.go - * @Description: 这是默认设置,请设置`customMade`, 打开koroFileHeader查看配置 进行设置: https://github.com/OBKoro1/koro1FileHeader/wiki/%E9%85%8D%E7%BD%AE - */ -package driver - -type Additional interface{} - -type Select string - -type Item struct { - Name string `json:"name"` - Type string `json:"type"` - Default string `json:"default"` - Options string `json:"options"` - Required bool `json:"required"` - Help string `json:"help"` -} - -type Info struct { - Common []Item `json:"common"` - Additional []Item `json:"additional"` - Config Config `json:"config"` -} - -type IRootPath interface { - GetRootPath() string -} - -type IRootId interface { - GetRootId() string -} - -type RootPath struct { - RootFolderPath string `json:"root_folder_path"` -} - -type RootID struct { - RootFolderID string `json:"root_folder_id" omit:"true"` -} - -func (r RootPath) GetRootPath() string { - return r.RootFolderPath -} - -func (r *RootPath) SetRootPath(path string) { - r.RootFolderPath = path -} - -func (r RootID) GetRootId() string { - return r.RootFolderID -} diff --git a/internal/op/const.go b/internal/op/const.go deleted file mode 100644 index a465735..0000000 --- a/internal/op/const.go +++ /dev/null @@ -1,6 +0,0 @@ -package op - -const ( - WORK = "work" - RootName = "root" -) diff --git a/internal/op/driver.go b/internal/op/driver.go deleted file mode 100644 index ba75641..0000000 --- a/internal/op/driver.go +++ /dev/null @@ -1,173 +0,0 @@ -package op - -import ( - "reflect" - "strings" - - "github.com/IceWhaleTech/CasaOS/internal/conf" - - "github.com/IceWhaleTech/CasaOS/internal/driver" - "github.com/pkg/errors" -) - -type New func() driver.Driver - -var driverNewMap = map[string]New{} -var driverInfoMap = map[string][]driver.Item{} //driver.Info{} - -func RegisterDriver(driver New) { - // log.Infof("register driver: [%s]", config.Name) - tempDriver := driver() - tempConfig := tempDriver.Config() - registerDriverItems(tempConfig, tempDriver.GetAddition()) - driverNewMap[tempConfig.Name] = driver -} - -func GetDriverNew(name string) (New, error) { - n, ok := driverNewMap[name] - if !ok { - return nil, errors.Errorf("no driver named: %s", name) - } - return n, nil -} - -func GetDriverNames() []string { - var driverNames []string - for k := range driverInfoMap { - driverNames = append(driverNames, k) - } - return driverNames -} - -// func GetDriverInfoMap() map[string]driver.Info { -// return driverInfoMap -// } -func GetDriverInfoMap() map[string][]driver.Item { - return driverInfoMap -} -func registerDriverItems(config driver.Config, addition driver.Additional) { - // log.Debugf("addition of %s: %+v", config.Name, addition) - tAddition := reflect.TypeOf(addition) - for tAddition.Kind() == reflect.Pointer { - tAddition = tAddition.Elem() - } - //mainItems := getMainItems(config) - additionalItems := getAdditionalItems(tAddition, config.DefaultRoot) - driverInfoMap[config.Name] = additionalItems - // driver.Info{ - // Common: mainItems, - // Additional: additionalItems, - // Config: config, - // } -} - -func getMainItems(config driver.Config) []driver.Item { - items := []driver.Item{{ - Name: "mount_path", - Type: conf.TypeString, - Required: true, - Help: "", - }, { - Name: "order", - Type: conf.TypeNumber, - Help: "use to sort", - }, { - Name: "remark", - Type: conf.TypeText, - }} - if !config.NoCache { - items = append(items, driver.Item{ - Name: "cache_expiration", - Type: conf.TypeNumber, - Default: "30", - Required: true, - Help: "The cache expiration time for this storage", - }) - } - if !config.OnlyProxy && !config.OnlyLocal { - items = append(items, []driver.Item{{ - Name: "web_proxy", - Type: conf.TypeBool, - }, { - Name: "webdav_policy", - Type: conf.TypeSelect, - Options: "302_redirect,use_proxy_url,native_proxy", - Default: "302_redirect", - Required: true, - }, - }...) - } else { - items = append(items, driver.Item{ - Name: "webdav_policy", - Type: conf.TypeSelect, - Default: "native_proxy", - Options: "use_proxy_url,native_proxy", - Required: true, - }) - } - items = append(items, driver.Item{ - Name: "down_proxy_url", - Type: conf.TypeText, - }) - if config.LocalSort { - items = append(items, []driver.Item{{ - Name: "order_by", - Type: conf.TypeSelect, - Options: "name,size,modified", - }, { - Name: "order_direction", - Type: conf.TypeSelect, - Options: "asc,desc", - }}...) - } - items = append(items, driver.Item{ - Name: "extract_folder", - Type: conf.TypeSelect, - Options: "front,back", - }) - return items -} - -func getAdditionalItems(t reflect.Type, defaultRoot string) []driver.Item { - var items []driver.Item - for i := 0; i < t.NumField(); i++ { - - field := t.Field(i) - if field.Type.Kind() == reflect.Struct { - items = append(items, getAdditionalItems(field.Type, defaultRoot)...) - continue - } - tag := field.Tag - ignore, ok1 := tag.Lookup("ignore") - name, ok2 := tag.Lookup("json") - if (ok1 && ignore == "true") || !ok2 { - continue - } - if tag.Get("omit") == "true" { - continue - } - item := driver.Item{ - Name: name, - Type: strings.ToLower(field.Type.Name()), - Default: tag.Get("default"), - Options: tag.Get("options"), - Required: tag.Get("required") == "true", - Help: tag.Get("help"), - } - if tag.Get("type") != "" { - item.Type = tag.Get("type") - } - if item.Name == "root_folder_id" || item.Name == "root_folder_path" { - if item.Default == "" { - item.Default = defaultRoot - } - item.Required = item.Default != "" - } - // set default type to string - if item.Type == "" { - item.Type = "string" - } - items = append(items, item) - } - return items -} diff --git a/internal/op/fs.go b/internal/op/fs.go deleted file mode 100644 index 576a4b6..0000000 --- a/internal/op/fs.go +++ /dev/null @@ -1,545 +0,0 @@ -package op - -import ( - "context" - "os" - stdpath "path" - "time" - - "github.com/IceWhaleTech/CasaOS/internal/driver" - "github.com/IceWhaleTech/CasaOS/model" - "github.com/IceWhaleTech/CasaOS/pkg/generic_sync" - "github.com/IceWhaleTech/CasaOS/pkg/singleflight" - "github.com/IceWhaleTech/CasaOS/pkg/utils" - "github.com/Xhofe/go-cache" - "github.com/pkg/errors" - pkgerr "github.com/pkg/errors" - log "github.com/sirupsen/logrus" -) - -// In order to facilitate adding some other things before and after file op - -var listCache = cache.NewMemCache(cache.WithShards[[]model.Obj](64)) -var listG singleflight.Group[[]model.Obj] - -func updateCacheObj(storage driver.Driver, path string, oldObj model.Obj, newObj model.Obj) { - key := Key(storage, path) - objs, ok := listCache.Get(key) - if ok { - for i, obj := range objs { - if obj.GetName() == oldObj.GetName() { - objs[i] = newObj - break - } - } - listCache.Set(key, objs, cache.WithEx[[]model.Obj](time.Minute*time.Duration(storage.GetStorage().CacheExpiration))) - } -} - -func delCacheObj(storage driver.Driver, path string, obj model.Obj) { - key := Key(storage, path) - objs, ok := listCache.Get(key) - if ok { - for i, oldObj := range objs { - if oldObj.GetName() == obj.GetName() { - objs = append(objs[:i], objs[i+1:]...) - break - } - } - listCache.Set(key, objs, cache.WithEx[[]model.Obj](time.Minute*time.Duration(storage.GetStorage().CacheExpiration))) - } -} - -var addSortDebounceMap generic_sync.MapOf[string, func(func())] - -func addCacheObj(storage driver.Driver, path string, newObj model.Obj) { - key := Key(storage, path) - objs, ok := listCache.Get(key) - if ok { - for i, obj := range objs { - if obj.GetName() == newObj.GetName() { - objs[i] = newObj - return - } - } - - // Simple separation of files and folders - if len(objs) > 0 && objs[len(objs)-1].IsDir() == newObj.IsDir() { - objs = append(objs, newObj) - } else { - objs = append([]model.Obj{newObj}, objs...) - } - - if storage.Config().LocalSort { - debounce, _ := addSortDebounceMap.LoadOrStore(key, utils.NewDebounce(time.Minute)) - log.Debug("addCacheObj: wait start sort") - debounce(func() { - log.Debug("addCacheObj: start sort") - model.SortFiles(objs, storage.GetStorage().OrderBy, storage.GetStorage().OrderDirection) - addSortDebounceMap.Delete(key) - }) - } - - listCache.Set(key, objs, cache.WithEx[[]model.Obj](time.Minute*time.Duration(storage.GetStorage().CacheExpiration))) - } -} - -func ClearCache(storage driver.Driver, path string) { - listCache.Del(Key(storage, path)) -} - -func Key(storage driver.Driver, path string) string { - return stdpath.Join(storage.GetStorage().MountPath, utils.FixAndCleanPath(path)) -} - -// List files in storage, not contains virtual file -func List(ctx context.Context, storage driver.Driver, path string, args model.ListArgs, refresh ...bool) ([]model.Obj, error) { - if storage.Config().CheckStatus && storage.GetStorage().Status != WORK { - return nil, errors.Errorf("storage not init: %s", storage.GetStorage().Status) - } - path = utils.FixAndCleanPath(path) - log.Debugf("op.List %s", path) - key := Key(storage, path) - if !utils.IsBool(refresh...) { - if files, ok := listCache.Get(key); ok { - log.Debugf("use cache when list %s", path) - return files, nil - } - } - dir, err := GetUnwrap(ctx, storage, path) - if err != nil { - return nil, errors.WithMessage(err, "failed get dir") - } - log.Debugf("list dir: %+v", dir) - if !dir.IsDir() { - return nil, errors.WithStack(errors.New("not a folder")) - } - objs, err, _ := listG.Do(key, func() ([]model.Obj, error) { - files, err := storage.List(ctx, dir, args) - if err != nil { - return nil, errors.Wrapf(err, "failed to list objs") - } - // set path - for _, f := range files { - if s, ok := f.(model.SetPath); ok && f.GetPath() == "" && dir.GetPath() != "" { - s.SetPath(stdpath.Join(dir.GetPath(), f.GetName())) - } - } - // warp obj name - model.WrapObjsName(files) - // call hooks - go func(reqPath string, files []model.Obj) { - for _, hook := range ObjsUpdateHooks { - hook(args.ReqPath, files) - } - }(args.ReqPath, files) - - // sort objs - if storage.Config().LocalSort { - model.SortFiles(files, storage.GetStorage().OrderBy, storage.GetStorage().OrderDirection) - } - model.ExtractFolder(files, storage.GetStorage().ExtractFolder) - - if !storage.Config().NoCache { - if len(files) > 0 { - log.Debugf("set cache: %s => %+v", key, files) - listCache.Set(key, files, cache.WithEx[[]model.Obj](time.Minute*time.Duration(storage.GetStorage().CacheExpiration))) - } else { - log.Debugf("del cache: %s", key) - listCache.Del(key) - } - } - return files, nil - }) - return objs, err -} - -// Get object from list of files -func Get(ctx context.Context, storage driver.Driver, path string) (model.Obj, error) { - path = utils.FixAndCleanPath(path) - log.Debugf("op.Get %s", path) - - // is root folder - if utils.PathEqual(path, "/") { - var rootObj model.Obj - switch r := storage.GetAddition().(type) { - case driver.IRootId: - rootObj = &model.Object{ - ID: r.GetRootId(), - Name: RootName, - Size: 0, - Modified: storage.GetStorage().Modified, - IsFolder: true, - Path: path, - } - case driver.IRootPath: - rootObj = &model.Object{ - Path: r.GetRootPath(), - Name: RootName, - Size: 0, - Modified: storage.GetStorage().Modified, - IsFolder: true, - } - default: - if storage, ok := storage.(driver.Getter); ok { - obj, err := storage.GetRoot(ctx) - if err != nil { - return nil, errors.WithMessage(err, "failed get root obj") - } - rootObj = obj - } - } - if rootObj == nil { - return nil, errors.Errorf("please implement IRootPath or IRootId or Getter method") - } - return &model.ObjWrapName{ - Name: RootName, - Obj: rootObj, - }, nil - } - - // not root folder - dir, name := stdpath.Split(path) - files, err := List(ctx, storage, dir, model.ListArgs{}) - if err != nil { - return nil, errors.WithMessage(err, "failed get parent list") - } - for _, f := range files { - // TODO maybe copy obj here - if f.GetName() == name { - return f, nil - } - } - log.Debugf("cant find obj with name: %s", name) - return nil, errors.WithStack(errors.New("object not found")) -} - -func GetUnwrap(ctx context.Context, storage driver.Driver, path string) (model.Obj, error) { - obj, err := Get(ctx, storage, path) - if err != nil { - return nil, err - } - return model.UnwrapObjs(obj), err -} - -var linkCache = cache.NewMemCache(cache.WithShards[*model.Link](16)) -var linkG singleflight.Group[*model.Link] - -// Link get link, if is an url. should have an expiry time -func Link(ctx context.Context, storage driver.Driver, path string, args model.LinkArgs) (*model.Link, model.Obj, error) { - if storage.Config().CheckStatus && storage.GetStorage().Status != WORK { - return nil, nil, errors.Errorf("storage not init: %s", storage.GetStorage().Status) - } - file, err := GetUnwrap(ctx, storage, path) - if err != nil { - return nil, nil, errors.WithMessage(err, "failed to get file") - } - if file.IsDir() { - return nil, nil, errors.WithStack(errors.New("not a file")) - } - key := Key(storage, path) + ":" + args.IP - if link, ok := linkCache.Get(key); ok { - return link, file, nil - } - fn := func() (*model.Link, error) { - link, err := storage.Link(ctx, file, args) - if err != nil { - return nil, errors.Wrapf(err, "failed get link") - } - if link.Expiration != nil { - linkCache.Set(key, link, cache.WithEx[*model.Link](*link.Expiration)) - } - return link, nil - } - link, err, _ := linkG.Do(key, fn) - return link, file, err -} - -// Other api -func Other(ctx context.Context, storage driver.Driver, args model.FsOtherArgs) (interface{}, error) { - obj, err := GetUnwrap(ctx, storage, args.Path) - if err != nil { - return nil, errors.WithMessagef(err, "failed to get obj") - } - if o, ok := storage.(driver.Other); ok { - return o.Other(ctx, model.OtherArgs{ - Obj: obj, - Method: args.Method, - Data: args.Data, - }) - } else { - return nil, errors.New("not implement") - } -} - -var mkdirG singleflight.Group[interface{}] - -func MakeDir(ctx context.Context, storage driver.Driver, path string, lazyCache ...bool) error { - if storage.Config().CheckStatus && storage.GetStorage().Status != WORK { - return errors.Errorf("storage not init: %s", storage.GetStorage().Status) - } - path = utils.FixAndCleanPath(path) - key := Key(storage, path) - _, err, _ := mkdirG.Do(key, func() (interface{}, error) { - // check if dir exists - f, err := GetUnwrap(ctx, storage, path) - if err != nil { - if errors.Is(pkgerr.Cause(err), errors.New("object not found")) { - parentPath, dirName := stdpath.Split(path) - err = MakeDir(ctx, storage, parentPath) - if err != nil { - return nil, errors.WithMessagef(err, "failed to make parent dir [%s]", parentPath) - } - parentDir, err := GetUnwrap(ctx, storage, parentPath) - // this should not happen - if err != nil { - return nil, errors.WithMessagef(err, "failed to get parent dir [%s]", parentPath) - } - - switch s := storage.(type) { - case driver.MkdirResult: - var newObj model.Obj - newObj, err = s.MakeDir(ctx, parentDir, dirName) - if err == nil { - if newObj != nil { - addCacheObj(storage, parentPath, model.WrapObjName(newObj)) - } else if !utils.IsBool(lazyCache...) { - ClearCache(storage, parentPath) - } - } - case driver.Mkdir: - err = s.MakeDir(ctx, parentDir, dirName) - if err == nil && !utils.IsBool(lazyCache...) { - ClearCache(storage, parentPath) - } - default: - return nil, errors.New("not implement") - } - return nil, errors.WithStack(err) - } - return nil, errors.WithMessage(err, "failed to check if dir exists") - } - // dir exists - if f.IsDir() { - return nil, nil - } - // dir to make is a file - return nil, errors.New("file exists") - }) - return err -} - -func Move(ctx context.Context, storage driver.Driver, srcPath, dstDirPath string, lazyCache ...bool) error { - if storage.Config().CheckStatus && storage.GetStorage().Status != WORK { - return errors.Errorf("storage not init: %s", storage.GetStorage().Status) - } - srcPath = utils.FixAndCleanPath(srcPath) - dstDirPath = utils.FixAndCleanPath(dstDirPath) - srcRawObj, err := Get(ctx, storage, srcPath) - if err != nil { - return errors.WithMessage(err, "failed to get src object") - } - srcObj := model.UnwrapObjs(srcRawObj) - dstDir, err := GetUnwrap(ctx, storage, dstDirPath) - if err != nil { - return errors.WithMessage(err, "failed to get dst dir") - } - srcDirPath := stdpath.Dir(srcPath) - - switch s := storage.(type) { - case driver.MoveResult: - var newObj model.Obj - newObj, err = s.Move(ctx, srcObj, dstDir) - if err == nil { - delCacheObj(storage, srcDirPath, srcRawObj) - if newObj != nil { - addCacheObj(storage, dstDirPath, model.WrapObjName(newObj)) - } else if !utils.IsBool(lazyCache...) { - ClearCache(storage, dstDirPath) - } - } - case driver.Move: - err = s.Move(ctx, srcObj, dstDir) - if err == nil { - delCacheObj(storage, srcDirPath, srcRawObj) - if !utils.IsBool(lazyCache...) { - ClearCache(storage, dstDirPath) - } - } - default: - return errors.New("not implement") - } - return errors.WithStack(err) -} - -func Rename(ctx context.Context, storage driver.Driver, srcPath, dstName string, lazyCache ...bool) error { - if storage.Config().CheckStatus && storage.GetStorage().Status != WORK { - return errors.Errorf("storage not init: %s", storage.GetStorage().Status) - } - srcPath = utils.FixAndCleanPath(srcPath) - srcRawObj, err := Get(ctx, storage, srcPath) - if err != nil { - return errors.WithMessage(err, "failed to get src object") - } - srcObj := model.UnwrapObjs(srcRawObj) - srcDirPath := stdpath.Dir(srcPath) - - switch s := storage.(type) { - case driver.RenameResult: - var newObj model.Obj - newObj, err = s.Rename(ctx, srcObj, dstName) - if err == nil { - if newObj != nil { - updateCacheObj(storage, srcDirPath, srcRawObj, model.WrapObjName(newObj)) - } else if !utils.IsBool(lazyCache...) { - ClearCache(storage, srcDirPath) - } - } - case driver.Rename: - err = s.Rename(ctx, srcObj, dstName) - if err == nil && !utils.IsBool(lazyCache...) { - ClearCache(storage, srcDirPath) - } - default: - return errors.New("not implement") - } - return errors.WithStack(err) -} - -// Copy Just copy file[s] in a storage -func Copy(ctx context.Context, storage driver.Driver, srcPath, dstDirPath string, lazyCache ...bool) error { - if storage.Config().CheckStatus && storage.GetStorage().Status != WORK { - return errors.Errorf("storage not init: %s", storage.GetStorage().Status) - } - srcPath = utils.FixAndCleanPath(srcPath) - dstDirPath = utils.FixAndCleanPath(dstDirPath) - srcObj, err := GetUnwrap(ctx, storage, srcPath) - if err != nil { - return errors.WithMessage(err, "failed to get src object") - } - dstDir, err := GetUnwrap(ctx, storage, dstDirPath) - if err != nil { - return errors.WithMessage(err, "failed to get dst dir") - } - - switch s := storage.(type) { - case driver.CopyResult: - var newObj model.Obj - newObj, err = s.Copy(ctx, srcObj, dstDir) - if err == nil { - if newObj != nil { - addCacheObj(storage, dstDirPath, model.WrapObjName(newObj)) - } else if !utils.IsBool(lazyCache...) { - ClearCache(storage, dstDirPath) - } - } - case driver.Copy: - err = s.Copy(ctx, srcObj, dstDir) - if err == nil && !utils.IsBool(lazyCache...) { - ClearCache(storage, dstDirPath) - } - default: - return errors.New("not implement") - } - return errors.WithStack(err) -} - -func Remove(ctx context.Context, storage driver.Driver, path string) error { - if storage.Config().CheckStatus && storage.GetStorage().Status != WORK { - return errors.Errorf("storage not init: %s", storage.GetStorage().Status) - } - path = utils.FixAndCleanPath(path) - rawObj, err := Get(ctx, storage, path) - if err != nil { - // if object not found, it's ok - if errors.Is(pkgerr.Cause(err), errors.New("object not found")) { - return nil - } - return errors.WithMessage(err, "failed to get object") - } - dirPath := stdpath.Dir(path) - - switch s := storage.(type) { - case driver.Remove: - err = s.Remove(ctx, model.UnwrapObjs(rawObj)) - if err == nil { - delCacheObj(storage, dirPath, rawObj) - } - default: - return errors.New("not implement") - } - return errors.WithStack(err) -} - -func Put(ctx context.Context, storage driver.Driver, dstDirPath string, file *model.FileStream, up driver.UpdateProgress, lazyCache ...bool) error { - if storage.Config().CheckStatus && storage.GetStorage().Status != WORK { - return errors.Errorf("storage not init: %s", storage.GetStorage().Status) - } - defer func() { - if f, ok := file.GetReadCloser().(*os.File); ok { - err := os.RemoveAll(f.Name()) - if err != nil { - log.Errorf("failed to remove file [%s]", f.Name()) - } - } - }() - defer func() { - if err := file.Close(); err != nil { - log.Errorf("failed to close file streamer, %v", err) - } - }() - // if file exist and size = 0, delete it - dstDirPath = utils.FixAndCleanPath(dstDirPath) - dstPath := stdpath.Join(dstDirPath, file.GetName()) - fi, err := GetUnwrap(ctx, storage, dstPath) - if err == nil { - if fi.GetSize() == 0 { - err = Remove(ctx, storage, dstPath) - if err != nil { - return errors.WithMessagef(err, "failed remove file that exist and have size 0") - } - } else { - file.Old = fi - } - } - err = MakeDir(ctx, storage, dstDirPath) - if err != nil { - return errors.WithMessagef(err, "failed to make dir [%s]", dstDirPath) - } - parentDir, err := GetUnwrap(ctx, storage, dstDirPath) - // this should not happen - if err != nil { - return errors.WithMessagef(err, "failed to get dir [%s]", dstDirPath) - } - // if up is nil, set a default to prevent panic - if up == nil { - up = func(p int) {} - } - - switch s := storage.(type) { - case driver.PutResult: - var newObj model.Obj - newObj, err = s.Put(ctx, parentDir, file, up) - if err == nil { - if newObj != nil { - addCacheObj(storage, dstDirPath, model.WrapObjName(newObj)) - } else if !utils.IsBool(lazyCache...) { - ClearCache(storage, dstDirPath) - } - } - case driver.Put: - err = s.Put(ctx, parentDir, file, up) - if err == nil && !utils.IsBool(lazyCache...) { - ClearCache(storage, dstDirPath) - } - default: - return errors.New("not implement") - } - log.Debugf("put file [%s] done", file.GetName()) - //if err == nil { - // //clear cache - // key := stdpath.Join(storage.GetStorage().MountPath, dstDirPath) - // listCache.Del(key) - //} - return errors.WithStack(err) -} diff --git a/internal/op/hook.go b/internal/op/hook.go deleted file mode 100644 index 5c07fc4..0000000 --- a/internal/op/hook.go +++ /dev/null @@ -1,109 +0,0 @@ -package op - -import ( - "regexp" - "strings" - - "github.com/IceWhaleTech/CasaOS-Common/utils/logger" - "github.com/IceWhaleTech/CasaOS/internal/conf" - "github.com/IceWhaleTech/CasaOS/internal/driver" - "github.com/IceWhaleTech/CasaOS/model" - jsoniter "github.com/json-iterator/go" - "github.com/pkg/errors" - "go.uber.org/zap" -) - -// Obj -type ObjsUpdateHook = func(parent string, objs []model.Obj) - -var ( - ObjsUpdateHooks = make([]ObjsUpdateHook, 0) -) - -func RegisterObjsUpdateHook(hook ObjsUpdateHook) { - ObjsUpdateHooks = append(ObjsUpdateHooks, hook) -} - -func HandleObjsUpdateHook(parent string, objs []model.Obj) { - for _, hook := range ObjsUpdateHooks { - hook(parent, objs) - } -} - -// Setting -type SettingItemHook func(item *model.SettingItem) error - -var settingItemHooks = map[string]SettingItemHook{ - conf.VideoTypes: func(item *model.SettingItem) error { - conf.SlicesMap[conf.VideoTypes] = strings.Split(item.Value, ",") - return nil - }, - conf.AudioTypes: func(item *model.SettingItem) error { - conf.SlicesMap[conf.AudioTypes] = strings.Split(item.Value, ",") - return nil - }, - conf.ImageTypes: func(item *model.SettingItem) error { - conf.SlicesMap[conf.ImageTypes] = strings.Split(item.Value, ",") - return nil - }, - conf.TextTypes: func(item *model.SettingItem) error { - conf.SlicesMap[conf.TextTypes] = strings.Split(item.Value, ",") - return nil - }, - conf.ProxyTypes: func(item *model.SettingItem) error { - conf.SlicesMap[conf.ProxyTypes] = strings.Split(item.Value, ",") - return nil - }, - conf.ProxyIgnoreHeaders: func(item *model.SettingItem) error { - conf.SlicesMap[conf.ProxyIgnoreHeaders] = strings.Split(item.Value, ",") - return nil - }, - conf.PrivacyRegs: func(item *model.SettingItem) error { - regStrs := strings.Split(item.Value, "\n") - regs := make([]*regexp.Regexp, 0, len(regStrs)) - for _, regStr := range regStrs { - reg, err := regexp.Compile(regStr) - if err != nil { - return errors.WithStack(err) - } - regs = append(regs, reg) - } - conf.PrivacyReg = regs - return nil - }, - conf.FilenameCharMapping: func(item *model.SettingItem) error { - var json = jsoniter.ConfigCompatibleWithStandardLibrary - err := json.UnmarshalFromString(item.Value, &conf.FilenameCharMap) - if err != nil { - return err - } - logger.Info("filename char mapping", zap.Any("FilenameCharMap", conf.FilenameCharMap)) - return nil - }, -} - -func RegisterSettingItemHook(key string, hook SettingItemHook) { - settingItemHooks[key] = hook -} - -func HandleSettingItemHook(item *model.SettingItem) (hasHook bool, err error) { - if hook, ok := settingItemHooks[item.Key]; ok { - return true, hook(item) - } - return false, nil -} - -// Storage -type StorageHook func(typ string, storage driver.Driver) - -var storageHooks = make([]StorageHook, 0) - -func CallStorageHooks(typ string, storage driver.Driver) { - for _, hook := range storageHooks { - hook(typ, storage) - } -} - -func RegisterStorageHook(hook StorageHook) { - storageHooks = append(storageHooks, hook) -} diff --git a/internal/sign/sign.go b/internal/sign/sign.go deleted file mode 100644 index 412bdfe..0000000 --- a/internal/sign/sign.go +++ /dev/null @@ -1,36 +0,0 @@ -package sign - -import ( - "sync" - "time" - - "github.com/IceWhaleTech/CasaOS/pkg/sign" -) - -var once sync.Once -var instance sign.Sign - -func Sign(data string) string { - - return NotExpired(data) - -} - -func WithDuration(data string, d time.Duration) string { - once.Do(Instance) - return instance.Sign(data, time.Now().Add(d).Unix()) -} - -func NotExpired(data string) string { - once.Do(Instance) - return instance.Sign(data, 0) -} - -func Verify(data string, sign string) error { - once.Do(Instance) - return instance.Verify(data, sign) -} - -func Instance() { - instance = sign.NewHMACSign([]byte("token")) -} diff --git a/main.go b/main.go index c169f51..b99161a 100644 --- a/main.go +++ b/main.go @@ -30,7 +30,6 @@ import ( "github.com/coreos/go-systemd/daemon" "go.uber.org/zap" - _ "github.com/IceWhaleTech/CasaOS/drivers" "github.com/robfig/cron" "gorm.io/gorm" ) @@ -79,7 +78,7 @@ func init() { service.Cache = cache.Init() service.GetCPUThermalZone() - service.MyService.Storages().InitStorages() + route.InitFunction() } @@ -95,6 +94,7 @@ func init() { // @name Authorization // @BasePath /v1 func main() { + if *versionFlag { return } @@ -141,9 +141,9 @@ func main() { "/v1/image", "/v1/samba", "/v1/notify", - "/v1/driver", - "/v1/cloud", - "/v1/recover", + //"/v1/driver", + //"/v1/cloud", + //"/v1/recover", "/v1/other", route.V2APIPath, route.V2DocPath, @@ -162,7 +162,6 @@ func main() { } var events []message_bus.EventType events = append(events, message_bus.EventType{Name: "casaos:system:utilization", SourceID: common.SERVICENAME, PropertyTypeList: []message_bus.PropertyType{}}) - events = append(events, message_bus.EventType{Name: "casaos:file:recover", SourceID: common.SERVICENAME, PropertyTypeList: []message_bus.PropertyType{}}) events = append(events, message_bus.EventType{Name: "casaos:file:operate", SourceID: common.SERVICENAME, PropertyTypeList: []message_bus.PropertyType{}}) // register at message bus for i := 0; i < 10; i++ { @@ -225,7 +224,7 @@ func main() { } logger.Info("CasaOS main service is listening...", zap.Any("address", listener.Addr().String())) - + //defer service.MyService.Storage().UnmountAllStorage() err = s.Serve(listener) // not using http.serve() to fix G114: Use of net/http serve function that has no support for setting timeouts (see https://github.com/securego/gosec) if err != nil { panic(err) diff --git a/pkg/mount/dir.go b/pkg/mount/dir.go new file mode 100644 index 0000000..3a484c1 --- /dev/null +++ b/pkg/mount/dir.go @@ -0,0 +1,268 @@ +package mount + +import ( + "context" + "fmt" + "io" + "os" + "syscall" + "time" + + "bazil.org/fuse" + fusefs "bazil.org/fuse/fs" + "github.com/rclone/rclone/cmd/mountlib" + "github.com/rclone/rclone/fs" + "github.com/rclone/rclone/fs/log" + "github.com/rclone/rclone/vfs" +) + +// Dir represents a directory entry +type Dir struct { + *vfs.Dir + fsys *FS +} + +// Check interface satisfied +var _ fusefs.Node = (*Dir)(nil) + +// Attr updates the attributes of a directory +func (d *Dir) Attr(ctx context.Context, a *fuse.Attr) (err error) { + defer log.Trace(d, "")("attr=%+v, err=%v", a, &err) + a.Valid = d.fsys.opt.AttrTimeout + a.Gid = d.VFS().Opt.GID + a.Uid = d.VFS().Opt.UID + a.Mode = os.ModeDir | d.VFS().Opt.DirPerms + modTime := d.ModTime() + a.Atime = modTime + a.Mtime = modTime + a.Ctime = modTime + // FIXME include Valid so get some caching? + // FIXME fs.Debugf(d.path, "Dir.Attr %+v", a) + return nil +} + +// Check interface satisfied +var _ fusefs.NodeSetattrer = (*Dir)(nil) + +// Setattr handles attribute changes from FUSE. Currently supports ModTime only. +func (d *Dir) Setattr(ctx context.Context, req *fuse.SetattrRequest, resp *fuse.SetattrResponse) (err error) { + defer log.Trace(d, "stat=%+v", req)("err=%v", &err) + if d.VFS().Opt.NoModTime { + return nil + } + + if req.Valid.MtimeNow() { + err = d.SetModTime(time.Now()) + } else if req.Valid.Mtime() { + err = d.SetModTime(req.Mtime) + } + + return err +} + +// Check interface satisfied +var _ fusefs.NodeRequestLookuper = (*Dir)(nil) + +// Lookup looks up a specific entry in the receiver. +// +// Lookup should return a Node corresponding to the entry. If the +// name does not exist in the directory, Lookup should return ENOENT. +// +// Lookup need not to handle the names "." and "..". +func (d *Dir) Lookup(ctx context.Context, req *fuse.LookupRequest, resp *fuse.LookupResponse) (node fusefs.Node, err error) { + defer log.Trace(d, "name=%q", req.Name)("node=%+v, err=%v", &node, &err) + mnode, err := d.Dir.Stat(req.Name) + if err != nil { + return nil, err + } + resp.EntryValid = d.fsys.opt.AttrTimeout + // Check the mnode to see if it has a fuse Node cached + // We must return the same fuse nodes for vfs Nodes + node, ok := mnode.Sys().(fusefs.Node) + if ok { + return node, nil + } + switch x := mnode.(type) { + case *vfs.File: + node = &File{x, d.fsys} + case *vfs.Dir: + node = &Dir{x, d.fsys} + default: + panic("bad type") + } + // Cache the node for later + mnode.SetSys(node) + return node, nil +} + +// Check interface satisfied +var _ fusefs.HandleReadDirAller = (*Dir)(nil) + +// ReadDirAll reads the contents of the directory +func (d *Dir) ReadDirAll(ctx context.Context) (dirents []fuse.Dirent, err error) { + itemsRead := -1 + defer log.Trace(d, "")("item=%d, err=%v", &itemsRead, &err) + items, err := d.Dir.ReadDirAll() + if err != nil { + return nil, (err) + } + dirents = append(dirents, fuse.Dirent{ + Type: fuse.DT_Dir, + Name: ".", + }, fuse.Dirent{ + Type: fuse.DT_Dir, + Name: "..", + }) + for _, node := range items { + name := node.Name() + if len(name) > mountlib.MaxLeafSize { + fs.Errorf(d, "Name too long (%d bytes) for FUSE, skipping: %s", len(name), name) + continue + } + var dirent = fuse.Dirent{ + // Inode FIXME ??? + Type: fuse.DT_File, + Name: name, + } + if node.IsDir() { + dirent.Type = fuse.DT_Dir + } + dirents = append(dirents, dirent) + } + itemsRead = len(dirents) + return dirents, nil +} + +var _ fusefs.NodeCreater = (*Dir)(nil) + +// Create makes a new file +func (d *Dir) Create(ctx context.Context, req *fuse.CreateRequest, resp *fuse.CreateResponse) (node fusefs.Node, handle fusefs.Handle, err error) { + defer log.Trace(d, "name=%q", req.Name)("node=%v, handle=%v, err=%v", &node, &handle, &err) + file, err := d.Dir.Create(req.Name, int(req.Flags)) + if err != nil { + return nil, nil, (err) + } + fh, err := file.Open(int(req.Flags) | os.O_CREATE) + if err != nil { + return nil, nil, (err) + } + node = &File{file, d.fsys} + file.SetSys(node) // cache the FUSE node for later + return node, &FileHandle{fh}, err +} + +var _ fusefs.NodeMkdirer = (*Dir)(nil) + +// Mkdir creates a new directory +func (d *Dir) Mkdir(ctx context.Context, req *fuse.MkdirRequest) (node fusefs.Node, err error) { + defer log.Trace(d, "name=%q", req.Name)("node=%+v, err=%v", &node, &err) + dir, err := d.Dir.Mkdir(req.Name) + if err != nil { + return nil, (err) + } + node = &Dir{dir, d.fsys} + dir.SetSys(node) // cache the FUSE node for later + return node, nil +} + +var _ fusefs.NodeRemover = (*Dir)(nil) + +// Remove removes the entry with the given name from +// the receiver, which must be a directory. The entry to be removed +// may correspond to a file (unlink) or to a directory (rmdir). +func (d *Dir) Remove(ctx context.Context, req *fuse.RemoveRequest) (err error) { + defer log.Trace(d, "name=%q", req.Name)("err=%v", &err) + err = d.Dir.RemoveName(req.Name) + if err != nil { + return (err) + } + return nil +} + +// Invalidate a leaf in a directory +func (d *Dir) invalidateEntry(dirNode fusefs.Node, leaf string) { + fs.Debugf(dirNode, "Invalidating %q", leaf) + err := d.fsys.server.InvalidateEntry(dirNode, leaf) + if err != nil { + fs.Debugf(dirNode, "Failed to invalidate %q: %v", leaf, err) + } +} + +// Check interface satisfied +var _ fusefs.NodeRenamer = (*Dir)(nil) + +// Rename the file +func (d *Dir) Rename(ctx context.Context, req *fuse.RenameRequest, newDir fusefs.Node) (err error) { + defer log.Trace(d, "oldName=%q, newName=%q, newDir=%+v", req.OldName, req.NewName, newDir)("err=%v", &err) + destDir, ok := newDir.(*Dir) + if !ok { + return fmt.Errorf("unknown Dir type %T", newDir) + } + + err = d.Dir.Rename(req.OldName, req.NewName, destDir.Dir) + if err != nil { + return (err) + } + + // Invalidate the new directory entry so it gets re-read (in + // the background otherwise we cause a deadlock) + // + // See https://github.com/rclone/rclone/issues/4977 for why + go d.invalidateEntry(newDir, req.NewName) + //go d.invalidateEntry(d, req.OldName) + + return nil +} + +// Check interface satisfied +var _ fusefs.NodeFsyncer = (*Dir)(nil) + +// Fsync the directory +func (d *Dir) Fsync(ctx context.Context, req *fuse.FsyncRequest) (err error) { + defer log.Trace(d, "")("err=%v", &err) + err = d.Dir.Sync() + if err != nil { + return (err) + } + return nil +} + +// Check interface satisfied +var _ fusefs.NodeLinker = (*Dir)(nil) + +// Link creates a new directory entry in the receiver based on an +// existing Node. Receiver must be a directory. +func (d *Dir) Link(ctx context.Context, req *fuse.LinkRequest, old fusefs.Node) (newNode fusefs.Node, err error) { + defer log.Trace(d, "req=%v, old=%v", req, old)("new=%v, err=%v", &newNode, &err) + return nil, syscall.ENOSYS +} + +// Check interface satisfied +var _ fusefs.NodeMknoder = (*Dir)(nil) + +// Mknod is called to create a file. Since we define create this will +// be called in preference, however NFS likes to call it for some +// reason. We don't actually create a file here just the Node. +func (d *Dir) Mknod(ctx context.Context, req *fuse.MknodRequest) (node fusefs.Node, err error) { + defer log.Trace(d, "name=%v, mode=%d, rdev=%d", req.Name, req.Mode, req.Rdev)("node=%v, err=%v", &node, &err) + if req.Rdev != 0 { + fs.Errorf(d, "Can't create device node %q", req.Name) + return nil, fuse.EIO + } + var cReq = fuse.CreateRequest{ + Name: req.Name, + Flags: fuse.OpenFlags(os.O_CREATE | os.O_WRONLY), + Mode: req.Mode, + Umask: req.Umask, + } + var cResp fuse.CreateResponse + node, handle, err := d.Create(ctx, &cReq, &cResp) + if err != nil { + return nil, err + } + err = handle.(io.Closer).Close() + if err != nil { + return nil, err + } + return node, nil +} diff --git a/pkg/mount/file.go b/pkg/mount/file.go new file mode 100644 index 0000000..ca09697 --- /dev/null +++ b/pkg/mount/file.go @@ -0,0 +1,125 @@ +package mount + +import ( + "context" + "syscall" + "time" + + "bazil.org/fuse" + fusefs "bazil.org/fuse/fs" + "github.com/rclone/rclone/fs/log" + "github.com/rclone/rclone/vfs" +) + +// File represents a file +type File struct { + *vfs.File + fsys *FS +} + +// Check interface satisfied +var _ fusefs.Node = (*File)(nil) + +// Attr fills out the attributes for the file +func (f *File) Attr(ctx context.Context, a *fuse.Attr) (err error) { + defer log.Trace(f, "")("a=%+v, err=%v", a, &err) + a.Valid = f.fsys.opt.AttrTimeout + modTime := f.File.ModTime() + Size := uint64(f.File.Size()) + Blocks := (Size + 511) / 512 + a.Gid = f.VFS().Opt.GID + a.Uid = f.VFS().Opt.UID + a.Mode = f.VFS().Opt.FilePerms + a.Size = Size + a.Atime = modTime + a.Mtime = modTime + a.Ctime = modTime + a.Blocks = Blocks + return nil +} + +// Check interface satisfied +var _ fusefs.NodeSetattrer = (*File)(nil) + +// Setattr handles attribute changes from FUSE. Currently supports ModTime and Size only +func (f *File) Setattr(ctx context.Context, req *fuse.SetattrRequest, resp *fuse.SetattrResponse) (err error) { + defer log.Trace(f, "a=%+v", req)("err=%v", &err) + if !f.VFS().Opt.NoModTime { + if req.Valid.Mtime() { + err = f.File.SetModTime(req.Mtime) + } else if req.Valid.MtimeNow() { + err = f.File.SetModTime(time.Now()) + } + } + if req.Valid.Size() { + err = f.File.Truncate(int64(req.Size)) + } + return (err) +} + +// Check interface satisfied +var _ fusefs.NodeOpener = (*File)(nil) + +// Open the file for read or write +func (f *File) Open(ctx context.Context, req *fuse.OpenRequest, resp *fuse.OpenResponse) (fh fusefs.Handle, err error) { + defer log.Trace(f, "flags=%v", req.Flags)("fh=%v, err=%v", &fh, &err) + + // fuse flags are based off syscall flags as are os flags, so + // should be compatible + handle, err := f.File.Open(int(req.Flags)) + if err != nil { + return nil, (err) + } + + // If size unknown then use direct io to read + if entry := handle.Node().DirEntry(); entry != nil && entry.Size() < 0 { + resp.Flags |= fuse.OpenDirectIO + } + + return &FileHandle{handle}, nil +} + +// Check interface satisfied +var _ fusefs.NodeFsyncer = (*File)(nil) + +// Fsync the file +// +// Note that we don't do anything except return OK +func (f *File) Fsync(ctx context.Context, req *fuse.FsyncRequest) (err error) { + defer log.Trace(f, "")("err=%v", &err) + return nil +} + +// Getxattr gets an extended attribute by the given name from the +// node. +// +// If there is no xattr by that name, returns fuse.ErrNoXattr. +func (f *File) Getxattr(ctx context.Context, req *fuse.GetxattrRequest, resp *fuse.GetxattrResponse) error { + return syscall.ENOSYS // we never implement this +} + +var _ fusefs.NodeGetxattrer = (*File)(nil) + +// Listxattr lists the extended attributes recorded for the node. +func (f *File) Listxattr(ctx context.Context, req *fuse.ListxattrRequest, resp *fuse.ListxattrResponse) error { + return syscall.ENOSYS // we never implement this +} + +var _ fusefs.NodeListxattrer = (*File)(nil) + +// Setxattr sets an extended attribute with the given name and +// value for the node. +func (f *File) Setxattr(ctx context.Context, req *fuse.SetxattrRequest) error { + return syscall.ENOSYS // we never implement this +} + +var _ fusefs.NodeSetxattrer = (*File)(nil) + +// Removexattr removes an extended attribute for the name. +// +// If there is no xattr by that name, returns fuse.ErrNoXattr. +func (f *File) Removexattr(ctx context.Context, req *fuse.RemovexattrRequest) error { + return syscall.ENOSYS // we never implement this +} + +var _ fusefs.NodeRemovexattrer = (*File)(nil) diff --git a/pkg/mount/handle.go b/pkg/mount/handle.go new file mode 100644 index 0000000..fce31cc --- /dev/null +++ b/pkg/mount/handle.go @@ -0,0 +1,82 @@ +package mount + +import ( + "context" + "io" + + "bazil.org/fuse" + fusefs "bazil.org/fuse/fs" + "github.com/rclone/rclone/fs/log" + "github.com/rclone/rclone/vfs" +) + +// FileHandle is an open for read file handle on a File +type FileHandle struct { + vfs.Handle +} + +// Check interface satisfied +var _ fusefs.HandleReader = (*FileHandle)(nil) + +// Read from the file handle +func (fh *FileHandle) Read(ctx context.Context, req *fuse.ReadRequest, resp *fuse.ReadResponse) (err error) { + var n int + defer log.Trace(fh, "len=%d, offset=%d", req.Size, req.Offset)("read=%d, err=%v", &n, &err) + data := make([]byte, req.Size) + n, err = fh.Handle.ReadAt(data, req.Offset) + if err == io.EOF { + err = nil + } else if err != nil { + return (err) + } + resp.Data = data[:n] + return nil +} + +// Check interface satisfied +var _ fusefs.HandleWriter = (*FileHandle)(nil) + +// Write data to the file handle +func (fh *FileHandle) Write(ctx context.Context, req *fuse.WriteRequest, resp *fuse.WriteResponse) (err error) { + defer log.Trace(fh, "len=%d, offset=%d", len(req.Data), req.Offset)("written=%d, err=%v", &resp.Size, &err) + n, err := fh.Handle.WriteAt(req.Data, req.Offset) + if err != nil { + return (err) + } + resp.Size = n + return nil +} + +// Check interface satisfied +var _ fusefs.HandleFlusher = (*FileHandle)(nil) + +// Flush is called on each close() of a file descriptor. So if a +// filesystem wants to return write errors in close() and the file has +// cached dirty data, this is a good place to write back data and +// return any errors. Since many applications ignore close() errors +// this is not always useful. +// +// NOTE: The flush() method may be called more than once for each +// open(). This happens if more than one file descriptor refers to an +// opened file due to dup(), dup2() or fork() calls. It is not +// possible to determine if a flush is final, so each flush should be +// treated equally. Multiple write-flush sequences are relatively +// rare, so this shouldn't be a problem. +// +// Filesystems shouldn't assume that flush will always be called after +// some writes, or that if will be called at all. +func (fh *FileHandle) Flush(ctx context.Context, req *fuse.FlushRequest) (err error) { + defer log.Trace(fh, "")("err=%v", &err) + return (fh.Handle.Flush()) +} + +var _ fusefs.HandleReleaser = (*FileHandle)(nil) + +// Release is called when we are finished with the file handle +// +// It isn't called directly from userspace so the error is ignored by +// the kernel +func (fh *FileHandle) Release(ctx context.Context, req *fuse.ReleaseRequest) (err error) { + defer log.Trace(fh, "")("err=%v", &err) + return (fh.Handle.Release()) +} diff --git a/pkg/mount/mount.go b/pkg/mount/mount.go new file mode 100644 index 0000000..9a14154 --- /dev/null +++ b/pkg/mount/mount.go @@ -0,0 +1,113 @@ +package mount + +import ( + "fmt" + + "bazil.org/fuse" + fusefs "bazil.org/fuse/fs" + "github.com/rclone/rclone/cmd/mountlib" + "github.com/rclone/rclone/fs" + "github.com/rclone/rclone/fs/log" + "github.com/rclone/rclone/vfs" +) + +func MountFn(VFS *vfs.VFS, mountpoint string, opt *mountlib.Options) (<-chan error, func() error, error) { + + f := VFS.Fs() + fs.Debugf(f, "Mounting on %q", mountpoint) + c, err := fuse.Mount(mountpoint, mountOptions(VFS, opt.DeviceName, opt)...) + if err != nil { + return nil, nil, err + } + filesys := NewFS(VFS, opt) + filesys.server = fusefs.New(c, nil) + + // Serve the mount point in the background returning error to errChan + errChan := make(chan error, 1) + go func() { + err := filesys.server.Serve(filesys) + closeErr := c.Close() + if err == nil { + err = closeErr + } + errChan <- err + }() + + unmount := func() error { + // Shutdown the VFS + filesys.VFS.Shutdown() + return fuse.Unmount(mountpoint) + } + return errChan, unmount, nil + +} +func NewFS(VFS *vfs.VFS, opt *mountlib.Options) *FS { + fsys := &FS{ + VFS: VFS, + f: VFS.Fs(), + opt: opt, + } + return fsys +} + +type FS struct { + *vfs.VFS + f fs.Fs + opt *mountlib.Options + server *fusefs.Server +} + +// mountOptions configures the options from the command line flags +func mountOptions(VFS *vfs.VFS, device string, opt *mountlib.Options) (options []fuse.MountOption) { + options = []fuse.MountOption{ + fuse.MaxReadahead(uint32(opt.MaxReadAhead)), + fuse.Subtype("rclone"), + fuse.FSName(device), + + // Options from benchmarking in the fuse module + //fuse.MaxReadahead(64 * 1024 * 1024), + //fuse.WritebackCache(), + } + if opt.AsyncRead { + options = append(options, fuse.AsyncRead()) + } + if opt.AllowNonEmpty { + options = append(options, fuse.AllowNonEmptyMount()) + } + if opt.AllowOther { + options = append(options, fuse.AllowOther()) + } + if opt.AllowRoot { + // options = append(options, fuse.AllowRoot()) + fs.Errorf(nil, "Ignoring --allow-root. Support has been removed upstream - see https://github.com/bazil/fuse/issues/144 for more info") + } + if opt.DefaultPermissions { + options = append(options, fuse.DefaultPermissions()) + } + if VFS.Opt.ReadOnly { + options = append(options, fuse.ReadOnly()) + } + if opt.WritebackCache { + options = append(options, fuse.WritebackCache()) + } + if opt.DaemonTimeout != 0 { + options = append(options, fuse.DaemonTimeout(fmt.Sprint(int(opt.DaemonTimeout.Seconds())))) + } + if len(opt.ExtraOptions) > 0 { + fs.Errorf(nil, "-o/--option not supported with this FUSE backend") + } + if len(opt.ExtraFlags) > 0 { + fs.Errorf(nil, "--fuse-flag not supported with this FUSE backend") + } + return options +} + +// Root returns the root node +func (f *FS) Root() (node fusefs.Node, err error) { + defer log.Trace("", "")("node=%+v, err=%v", &node, &err) + root, err := f.VFS.Root() + if err != nil { + return nil, err + } + return &Dir{root, f}, nil +} diff --git a/pkg/utils/httper/drive.go b/pkg/utils/httper/drive.go index 3d0ee30..b9200de 100644 --- a/pkg/utils/httper/drive.go +++ b/pkg/utils/httper/drive.go @@ -1,24 +1,17 @@ package httper import ( - "encoding/json" - "fmt" - "net" - "net/http" "time" - - "github.com/IceWhaleTech/CasaOS-Common/utils/logger" - "github.com/go-resty/resty/v2" - "go.uber.org/zap" ) type MountList struct { - MountPoints []struct { - MountPoint string `json:"MountPoint"` - Fs string `json:"Fs"` - Icon string `json:"Icon"` - Name string `json:"Name"` - } `json:"mountPoints"` + MountPoints []MountPoints `json:"mountPoints"` +} +type MountPoints struct { + MountPoint string `json:"MountPoint"` + Fs string `json:"Fs"` + Icon string `json:"Icon"` + Name string `json:"Name"` } type MountPoint struct { MountPoint string `json:"mount_point"` @@ -43,125 +36,126 @@ type RemotesResult struct { var UserAgent = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/87.0.4280.88 Safari/537.36" var DefaultTimeout = time.Second * 30 -func NewRestyClient() *resty.Client { +// func NewRestyClient() *resty.Client { - unixSocket := "/var/run/rclone/rclone.sock" +// unixSocket := "/var/run/rclone/rclone.sock" - transport := http.Transport{ - Dial: func(_, _ string) (net.Conn, error) { - return net.Dial("unix", unixSocket) - }, - } +// transport := http.Transport{ +// Dial: func(_, _ string) (net.Conn, error) { +// return net.Dial("unix", unixSocket) +// }, +// } - client := resty.New() +// client := resty.New() - client.SetTransport(&transport).SetBaseURL("http://localhost") - client.SetRetryCount(3).SetRetryWaitTime(5*time.Second).SetTimeout(DefaultTimeout).SetHeader("User-Agent", UserAgent) - return client -} +// client.SetTransport(&transport).SetBaseURL("http://localhost") +// client.SetRetryCount(3).SetRetryWaitTime(5*time.Second).SetTimeout(DefaultTimeout).SetHeader("User-Agent", UserAgent) +// return client +// } -func GetMountList() (MountList, error) { - var result MountList - res, err := NewRestyClient().R().Post("/mount/listmounts") - if err != nil { - return result, err - } - if res.StatusCode() != 200 { - return result, fmt.Errorf("get mount list failed") - } - json.Unmarshal(res.Body(), &result) - for i := 0; i < len(result.MountPoints); i++ { - result.MountPoints[i].Fs = result.MountPoints[i].Fs[:len(result.MountPoints[i].Fs)-1] - } - return result, err -} -func Mount(mountPoint string, fs string) error { - res, err := NewRestyClient().R().SetFormData(map[string]string{ - "mountPoint": mountPoint, - "fs": fs, - "mountOpt": `{"AllowOther": true}`, - }).Post("/mount/mount") - if err != nil { - return err - } - if res.StatusCode() != 200 { - return fmt.Errorf("mount failed") - } - logger.Info("mount then", zap.Any("res", res.Body())) - return nil -} -func Unmount(mountPoint string) error { - res, err := NewRestyClient().R().SetFormData(map[string]string{ - "mountPoint": mountPoint, - }).Post("/mount/unmount") - if err != nil { - logger.Error("when unmount", zap.Error(err)) - return err - } - if res.StatusCode() != 200 { - logger.Error("then unmount failed", zap.Any("res", res.Body())) - return fmt.Errorf("unmount failed") - } - logger.Info("unmount then", zap.Any("res", res.Body())) - return nil -} +// func GetMountList() (MountList, error) { +// var result MountList +// res, err := NewRestyClient().R().Post("/mount/listmounts") +// if err != nil { +// return result, err +// } +// if res.StatusCode() != 200 { +// return result, fmt.Errorf("get mount list failed") +// } +// json.Unmarshal(res.Body(), &result) +// for i := 0; i < len(result.MountPoints); i++ { +// result.MountPoints[i].Fs = result.MountPoints[i].Fs[:len(result.MountPoints[i].Fs)-1] +// } +// return result, err +// } +// +// func Mount(mountPoint string, fs string) error { +// res, err := NewRestyClient().R().SetFormData(map[string]string{ +// "mountPoint": mountPoint, +// "fs": fs, +// "mountOpt": `{"AllowOther": true}`, +// }).Post("/mount/mount") +// if err != nil { +// return err +// } +// if res.StatusCode() != 200 { +// return fmt.Errorf("mount failed") +// } +// logger.Info("mount then", zap.Any("res", res.Body())) +// return nil +// } +// func Unmount(mountPoint string) error { +// res, err := NewRestyClient().R().SetFormData(map[string]string{ +// "mountPoint": mountPoint, +// }).Post("/mount/unmount") +// if err != nil { +// logger.Error("when unmount", zap.Error(err)) +// return err +// } +// if res.StatusCode() != 200 { +// logger.Error("then unmount failed", zap.Any("res", res.Body())) +// return fmt.Errorf("unmount failed") +// } +// logger.Info("unmount then", zap.Any("res", res.Body())) +// return nil +// } -func CreateConfig(data map[string]string, name, t string) error { - data["config_is_local"] = "false" - dataStr, _ := json.Marshal(data) - res, err := NewRestyClient().R().SetFormData(map[string]string{ - "name": name, - "parameters": string(dataStr), - "type": t, - }).Post("/config/create") - logger.Info("when create config then", zap.Any("res", res.Body())) - if err != nil { - return err - } - if res.StatusCode() != 200 { - return fmt.Errorf("create config failed") - } +// func CreateConfig(data map[string]string, name, t string) error { +// data["config_is_local"] = "false" +// dataStr, _ := json.Marshal(data) +// res, err := NewRestyClient().R().SetFormData(map[string]string{ +// "name": name, +// "parameters": string(dataStr), +// "type": t, +// }).Post("/config/create") +// logger.Info("when create config then", zap.Any("res", res.Body())) +// if err != nil { +// return err +// } +// if res.StatusCode() != 200 { +// return fmt.Errorf("create config failed") +// } - return nil -} +// return nil +// } -func GetConfigByName(name string) (map[string]string, error) { +// func GetConfigByName(name string) (map[string]string, error) { - res, err := NewRestyClient().R().SetFormData(map[string]string{ - "name": name, - }).Post("/config/get") - if err != nil { - return nil, err - } - if res.StatusCode() != 200 { - return nil, fmt.Errorf("create config failed") - } - var result map[string]string - json.Unmarshal(res.Body(), &result) - return result, nil -} -func GetAllConfigName() (RemotesResult, error) { - var result RemotesResult - res, err := NewRestyClient().R().SetFormData(map[string]string{}).Post("/config/listremotes") - if err != nil { - return result, err - } - if res.StatusCode() != 200 { - return result, fmt.Errorf("get config failed") - } +// res, err := NewRestyClient().R().SetFormData(map[string]string{ +// "name": name, +// }).Post("/config/get") +// if err != nil { +// return nil, err +// } +// if res.StatusCode() != 200 { +// return nil, fmt.Errorf("create config failed") +// } +// var result map[string]string +// json.Unmarshal(res.Body(), &result) +// return result, nil +// } +// func GetAllConfigName() (RemotesResult, error) { +// var result RemotesResult +// res, err := NewRestyClient().R().SetFormData(map[string]string{}).Post("/config/listremotes") +// if err != nil { +// return result, err +// } +// if res.StatusCode() != 200 { +// return result, fmt.Errorf("get config failed") +// } - json.Unmarshal(res.Body(), &result) - return result, nil -} -func DeleteConfigByName(name string) error { - res, err := NewRestyClient().R().SetFormData(map[string]string{ - "name": name, - }).Post("/config/delete") - if err != nil { - return err - } - if res.StatusCode() != 200 { - return fmt.Errorf("delete config failed") - } - return nil -} +// json.Unmarshal(res.Body(), &result) +// return result, nil +// } +// func DeleteConfigByName(name string) error { +// res, err := NewRestyClient().R().SetFormData(map[string]string{ +// "name": name, +// }).Post("/config/delete") +// if err != nil { +// return err +// } +// if res.StatusCode() != 200 { +// return fmt.Errorf("delete config failed") +// } +// return nil +// } diff --git a/route/init.go b/route/init.go index eee2426..441278f 100644 --- a/route/init.go +++ b/route/init.go @@ -93,9 +93,4 @@ func InitNetworkMount() { connection.Directories = strings.Join(directories, ",") service.MyService.Connections().UpdateConnection(&connection) } - - err := service.MyService.Storage().CheckAndMountAll() - if err != nil { - logger.Error("mount storage err", zap.Any("err", err)) - } } diff --git a/route/v1.go b/route/v1.go index 03245f2..ca344b1 100644 --- a/route/v1.go +++ b/route/v1.go @@ -36,7 +36,7 @@ func InitV1Router() *gin.Engine { r.GET("/ping", func(ctx *gin.Context) { ctx.String(200, "pong") }) - r.GET("/v1/recover/:type", v1.GetRecoverStorage) + v1Group := r.Group("/v1") v1Group.Use(jwt.ExceptLocalhost()) @@ -97,17 +97,7 @@ func InitV1Router() *gin.Engine { // v1FileGroup.GET("/download", v1.UserFileDownloadCommonService) } - v1CloudGroup := v1Group.Group("/cloud") - v1CloudGroup.Use() - { - v1CloudGroup.GET("", v1.ListStorages) - v1CloudGroup.DELETE("", v1.DeleteStorage) - } - v1DriverGroup := v1Group.Group("/driver") - v1DriverGroup.Use() - { - v1DriverGroup.GET("", v1.ListDriverInfo) - } + v1FolderGroup := v1Group.Group("/folder") v1FolderGroup.Use() { diff --git a/route/v1/driver.go b/route/v1/driver.go deleted file mode 100644 index 951f702..0000000 --- a/route/v1/driver.go +++ /dev/null @@ -1,12 +0,0 @@ -package v1 - -import ( - "github.com/IceWhaleTech/CasaOS/internal/op" - "github.com/IceWhaleTech/CasaOS/model" - "github.com/IceWhaleTech/CasaOS/pkg/utils/common_err" - "github.com/gin-gonic/gin" -) - -func ListDriverInfo(c *gin.Context) { - c.JSON(common_err.SUCCESS, model.Result{Success: common_err.SUCCESS, Message: common_err.GetMsg(common_err.SUCCESS), Data: op.GetDriverInfoMap()}) -} diff --git a/route/v1/file.go b/route/v1/file.go index 2564f10..29f1b90 100644 --- a/route/v1/file.go +++ b/route/v1/file.go @@ -1,7 +1,6 @@ package v1 import ( - "errors" "fmt" "io" "io/ioutil" @@ -15,18 +14,15 @@ import ( "strconv" "strings" "sync" + "time" "github.com/IceWhaleTech/CasaOS-Common/utils/logger" - "github.com/IceWhaleTech/CasaOS/internal/conf" - "github.com/IceWhaleTech/CasaOS/internal/driver" "github.com/IceWhaleTech/CasaOS/model" - "github.com/IceWhaleTech/CasaOS/pkg/utils" "github.com/IceWhaleTech/CasaOS/pkg/utils/common_err" "github.com/IceWhaleTech/CasaOS/pkg/utils/file" "github.com/IceWhaleTech/CasaOS/service" - "github.com/IceWhaleTech/CasaOS/internal/sign" "github.com/gin-gonic/gin" uuid "github.com/satori/go.uuid" "go.uber.org/zap" @@ -34,6 +30,33 @@ import ( "github.com/h2non/filetype" ) +type ListReq struct { + model.PageReq + Path string `json:"path" form:"path"` + //Refresh bool `json:"refresh"` +} +type ObjResp struct { + Name string `json:"name"` + Size int64 `json:"size"` + IsDir bool `json:"is_dir"` + Modified time.Time `json:"modified"` + Sign string `json:"sign"` + Thumb string `json:"thumb"` + Type int `json:"type"` + Path string `json:"path"` + Date time.Time `json:"date"` + Extensions map[string]interface{} `json:"extensions"` +} +type FsListResp struct { + Content []ObjResp `json:"content"` + Total int64 `json:"total"` + Readme string `json:"readme,omitempty"` + Write bool `json:"write,omitempty"` + Provider string `json:"provider,omitempty"` + Index int `json:"index"` + Size int `json:"size"` +} + // @Summary 读取文件 // @Produce application/json // @Accept application/json @@ -766,168 +789,3 @@ func GetSize(c *gin.Context) { } c.JSON(common_err.SUCCESS, model.Result{Success: common_err.SUCCESS, Message: common_err.GetMsg(common_err.SUCCESS), Data: size}) } -func Proxy(c *gin.Context) { - rawPath := c.Query("path") - filename := filepath.Base(rawPath) - storage, err := service.MyService.FsService().GetStorage(rawPath) - if err != nil { - c.JSON(500, model.Result{Success: common_err.SERVICE_ERROR, Message: common_err.GetMsg(common_err.SERVICE_ERROR), Data: err.Error()}) - return - } - if canProxy(storage, filename) { - downProxyUrl := storage.GetStorage().DownProxyUrl - if downProxyUrl != "" { - _, ok := c.GetQuery("d") - if !ok { - URL := fmt.Sprintf("%s%s?sign=%s", - strings.Split(downProxyUrl, "\n")[0], - utils.EncodePath(rawPath, true), - sign.Sign(rawPath)) - c.Redirect(302, URL) - return - } - } - link, file, err := service.MyService.FsService().Link(c, rawPath, model.LinkArgs{ - Header: c.Request.Header, - Type: c.Query("type"), - }) - if err != nil { - c.JSON(common_err.SERVICE_ERROR, model.Result{Success: common_err.SERVICE_ERROR, Message: common_err.GetMsg(common_err.SERVICE_ERROR), Data: err.Error()}) - - return - } - err = CommonProxy(c.Writer, c.Request, link, file) - if err != nil { - c.JSON(common_err.SERVICE_ERROR, model.Result{Success: common_err.SERVICE_ERROR, Message: common_err.GetMsg(common_err.SERVICE_ERROR), Data: err.Error()}) - return - } - } else { - c.JSON(common_err.SERVICE_ERROR, model.Result{Success: common_err.SERVICE_ERROR, Message: common_err.GetMsg(common_err.SERVICE_ERROR), Data: "proxy not allowed"}) - return - } -} - -// TODO need optimize -// when should be proxy? -// 1. config.MustProxy() -// 2. storage.WebProxy -// 3. proxy_types -func shouldProxy(storage driver.Driver, filename string) bool { - if storage.Config().MustProxy() || storage.GetStorage().WebProxy { - return true - } - if utils.SliceContains(conf.SlicesMap[conf.ProxyTypes], utils.Ext(filename)) { - return true - } - return false -} - -// TODO need optimize -// when can be proxy? -// 1. text file -// 2. config.MustProxy() -// 3. storage.WebProxy -// 4. proxy_types -// solution: text_file + shouldProxy() -func canProxy(storage driver.Driver, filename string) bool { - if storage.Config().MustProxy() || storage.GetStorage().WebProxy || storage.GetStorage().WebdavProxy() { - return true - } - if utils.SliceContains(conf.SlicesMap[conf.ProxyTypes], utils.Ext(filename)) { - return true - } - if utils.SliceContains(conf.SlicesMap[conf.TextTypes], utils.Ext(filename)) { - return true - } - return false -} - -var HttpClient = &http.Client{} - -func CommonProxy(w http.ResponseWriter, r *http.Request, link *model.Link, file model.Obj) error { - // read data with native - var err error - if link.Data != nil { - defer func() { - _ = link.Data.Close() - }() - w.Header().Set("Content-Type", "application/octet-stream") - w.Header().Set("Content-Disposition", fmt.Sprintf(`attachment; filename="%s"; filename*=UTF-8''%s`, file.GetName(), url.QueryEscape(file.GetName()))) - w.Header().Set("Content-Length", strconv.FormatInt(file.GetSize(), 10)) - if link.Header != nil { - // TODO clean header with blacklist or whitelist - link.Header.Del("set-cookie") - for h, val := range link.Header { - w.Header()[h] = val - } - } - if link.Status == 0 { - w.WriteHeader(http.StatusOK) - } else { - w.WriteHeader(link.Status) - } - _, err = io.Copy(w, link.Data) - if err != nil { - return err - } - return nil - } - // local file - if link.FilePath != nil && *link.FilePath != "" { - f, err := os.Open(*link.FilePath) - if err != nil { - return err - } - defer func() { - _ = f.Close() - }() - fileStat, err := os.Stat(*link.FilePath) - if err != nil { - return err - } - w.Header().Set("Content-Disposition", fmt.Sprintf(`attachment; filename="%s"; filename*=UTF-8''%s`, file.GetName(), url.QueryEscape(file.GetName()))) - http.ServeContent(w, r, file.GetName(), fileStat.ModTime(), f) - return nil - } else { - req, err := http.NewRequest(link.Method, link.URL, nil) - if err != nil { - return err - } - for h, val := range r.Header { - if utils.SliceContains(conf.SlicesMap[conf.ProxyIgnoreHeaders], strings.ToLower(h)) { - continue - } - req.Header[h] = val - } - for h, val := range link.Header { - req.Header[h] = val - } - res, err := HttpClient.Do(req) - if err != nil { - return err - } - defer func() { - _ = res.Body.Close() - }() - logger.Info("proxy status", zap.Any("status", res.StatusCode)) - // TODO clean header with blacklist or whitelist - res.Header.Del("set-cookie") - for h, v := range res.Header { - w.Header()[h] = v - } - w.WriteHeader(res.StatusCode) - w.Header().Set("Content-Disposition", fmt.Sprintf(`attachment; filename="%s"; filename*=UTF-8''%s`, file.GetName(), url.QueryEscape(file.GetName()))) - if res.StatusCode >= 400 { - all, _ := ioutil.ReadAll(res.Body) - msg := string(all) - logger.Info("msg", zap.Any("msg", msg)) - - return errors.New(msg) - } - _, err = io.Copy(w, res.Body) - if err != nil { - return err - } - return nil - } -} diff --git a/route/v1/file_read.go b/route/v1/file_read.go deleted file mode 100644 index 7324db5..0000000 --- a/route/v1/file_read.go +++ /dev/null @@ -1,97 +0,0 @@ -package v1 - -import ( - "path/filepath" - "time" - - "github.com/IceWhaleTech/CasaOS/model" - "github.com/IceWhaleTech/CasaOS/pkg/utils/common_err" - "github.com/IceWhaleTech/CasaOS/service" - - "github.com/gin-gonic/gin" -) - -type ListReq struct { - model.PageReq - Path string `json:"path" form:"path"` - //Refresh bool `json:"refresh"` -} -type ObjResp struct { - Name string `json:"name"` - Size int64 `json:"size"` - IsDir bool `json:"is_dir"` - Modified time.Time `json:"modified"` - Sign string `json:"sign"` - Thumb string `json:"thumb"` - Type int `json:"type"` - Path string `json:"path"` - Date time.Time `json:"date"` - Extensions map[string]interface{} `json:"extensions"` -} -type FsListResp struct { - Content []ObjResp `json:"content"` - Total int64 `json:"total"` - Readme string `json:"readme,omitempty"` - Write bool `json:"write,omitempty"` - Provider string `json:"provider,omitempty"` - Index int `json:"index"` - Size int `json:"size"` -} - -func FsList(c *gin.Context) { - var req ListReq - if err := c.ShouldBind(&req); err != nil { - c.JSON(common_err.SUCCESS, model.Result{Success: common_err.CLIENT_ERROR, Message: common_err.GetMsg(common_err.CLIENT_ERROR), Data: err.Error()}) - return - } - req.Validate() - objs, err := service.MyService.FsService().FList(c, req.Path, false) - if err != nil { - c.JSON(common_err.SUCCESS, model.Result{Success: common_err.SERVICE_ERROR, Message: common_err.GetMsg(common_err.SERVICE_ERROR), Data: err.Error()}) - return - } - total, objs := pagination(objs, &req.PageReq) - provider := "unknown" - storage, err := service.MyService.FsService().GetStorage(req.Path) - if err == nil { - provider = storage.GetStorage().Driver - } - c.JSON(common_err.SUCCESS, model.Result{Success: common_err.SUCCESS, Message: common_err.GetMsg(common_err.SUCCESS), Data: FsListResp{ - Content: toObjsResp(objs, req.Path, false), - Total: int64(total), - Readme: "", - Write: false, - Provider: provider, - }}) -} -func pagination(objs []model.Obj, req *model.PageReq) (int, []model.Obj) { - pageIndex, pageSize := req.Index, req.Size - total := len(objs) - start := (pageIndex - 1) * pageSize - if start > total { - return total, []model.Obj{} - } - end := start + pageSize - if end > total { - end = total - } - return total, objs[start:end] -} - -func toObjsResp(objs []model.Obj, parent string, encrypt bool) []ObjResp { - var resp []ObjResp - for _, obj := range objs { - thumb, _ := model.GetThumb(obj) - resp = append(resp, ObjResp{ - Name: obj.GetName(), - Size: obj.GetSize(), - IsDir: obj.IsDir(), - Modified: obj.ModTime(), - Sign: "", - Path: filepath.Join(parent, obj.GetName()), - Thumb: thumb, - Type: 0, - }) - } - return resp -} diff --git a/route/v1/recover.go b/route/v1/recover.go deleted file mode 100644 index 9133afe..0000000 --- a/route/v1/recover.go +++ /dev/null @@ -1,203 +0,0 @@ -package v1 - -import ( - "strconv" - "strings" - "time" - - "github.com/IceWhaleTech/CasaOS-Common/utils/logger" - "github.com/IceWhaleTech/CasaOS/drivers/dropbox" - "github.com/IceWhaleTech/CasaOS/drivers/google_drive" - "github.com/IceWhaleTech/CasaOS/service" - "github.com/gin-gonic/gin" - "go.uber.org/zap" -) - -func GetRecoverStorage(c *gin.Context) { - c.Header("Content-Type", "text/html; charset=utf-8") - t := c.Param("type") - currentTime := time.Now().UTC() - currentDate := time.Now().UTC().Format("2006-01-02") - notify := make(map[string]interface{}) - if t == "GoogleDrive" { - add := google_drive.Addition{} - add.Code = c.Query("code") - if len(add.Code) == 0 { - c.String(200, `

Code cannot be empty

`) - notify["status"] = "fail" - notify["message"] = "Code cannot be empty" - logger.Error("Then code is empty: ", zap.String("code", add.Code), zap.Any("name", "google_drive")) - service.MyService.Notify().SendNotify("casaos:file:recover", notify) - return - } - - add.RootFolderID = "root" - add.ClientID = google_drive.CLIENTID - add.ClientSecret = google_drive.CLIENTSECRET - - var google_drive google_drive.GoogleDrive - google_drive.Addition = add - err := google_drive.Init(c) - if err != nil { - c.String(200, `

Initialization failure:`+err.Error()+`

`) - notify["status"] = "fail" - notify["message"] = "Initialization failure" - logger.Error("Then init error: ", zap.Error(err), zap.Any("name", "google_drive")) - service.MyService.Notify().SendNotify("casaos:file:recover", notify) - return - } - - username, err := google_drive.GetUserInfo(c) - if err != nil { - c.String(200, `

Failed to get user information:`+err.Error()+`

`) - notify["status"] = "fail" - notify["message"] = "Failed to get user information" - logger.Error("Then get user info error: ", zap.Error(err), zap.Any("name", "google_drive")) - service.MyService.Notify().SendNotify("casaos:file:recover", notify) - return - } - dmap := make(map[string]string) - dmap["username"] = username - configs, err := service.MyService.Storage().GetConfig() - if err != nil { - c.String(200, `

Failed to get rclone config:`+err.Error()+`

`) - notify["status"] = "fail" - notify["message"] = "Failed to get rclone config" - logger.Error("Then get config error: ", zap.Error(err), zap.Any("name", "google_drive")) - service.MyService.Notify().SendNotify("casaos:file:recover", notify) - return - } - for _, v := range configs.Remotes { - cf, err := service.MyService.Storage().GetConfigByName(v) - if err != nil { - logger.Error("then get config by name error: ", zap.Error(err), zap.Any("name", v)) - continue - } - if cf["type"] == "drive" && cf["username"] == dmap["username"] { - c.String(200, `

The same configuration has been added

`) - err := service.MyService.Storage().CheckAndMountByName(v) - if err != nil { - logger.Error("check and mount by name error: ", zap.Error(err), zap.Any("name", cf["username"])) - } - notify["status"] = "warn" - notify["message"] = "The same configuration has been added" - service.MyService.Notify().SendNotify("casaos:file:recover", notify) - return - } - } - if len(username) > 0 { - a := strings.Split(username, "@") - username = a[0] - } - - //username = fileutil.NameAccumulation(username, "/mnt") - username += "_google_drive_" + strconv.FormatInt(time.Now().Unix(), 10) - - dmap["client_id"] = add.ClientID - dmap["client_secret"] = add.ClientSecret - dmap["scope"] = "drive" - dmap["mount_point"] = "/mnt/" + username - dmap["token"] = `{"access_token":"` + google_drive.AccessToken + `","token_type":"Bearer","refresh_token":"` + google_drive.RefreshToken + `","expiry":"` + currentDate + `T` + currentTime.Add(time.Hour*1).Add(time.Minute*50).Format("15:04:05") + `Z"}` - service.MyService.Storage().CreateConfig(dmap, username, "drive") - service.MyService.Storage().MountStorage("/mnt/"+username, username+":") - notify := make(map[string]interface{}) - notify["status"] = "success" - notify["message"] = "Success" - service.MyService.Notify().SendNotify("casaos:file:recover", notify) - } else if t == "Dropbox" { - add := dropbox.Addition{} - add.Code = c.Query("code") - if len(add.Code) == 0 { - c.String(200, `

Code cannot be empty

`) - notify["status"] = "fail" - notify["message"] = "Code cannot be empty" - logger.Error("Then code is empty error: ", zap.String("code", add.Code), zap.Any("name", "dropbox")) - service.MyService.Notify().SendNotify("casaos:file:recover", notify) - return - } - add.RootFolderID = "" - add.AppKey = dropbox.APPKEY - add.AppSecret = dropbox.APPSECRET - var dropbox dropbox.Dropbox - dropbox.Addition = add - err := dropbox.Init(c) - if err != nil { - c.String(200, `

Initialization failure:`+err.Error()+`

`) - notify["status"] = "fail" - notify["message"] = "Initialization failure" - logger.Error("Then init error: ", zap.Error(err), zap.Any("name", "dropbox")) - service.MyService.Notify().SendNotify("casaos:file:recover", notify) - return - } - username, err := dropbox.GetUserInfo(c) - if err != nil { - c.String(200, `

Failed to get user information:`+err.Error()+`

`) - notify["status"] = "fail" - notify["message"] = "Failed to get user information" - logger.Error("Then get user information: ", zap.Error(err), zap.Any("name", "dropbox")) - service.MyService.Notify().SendNotify("casaos:file:recover", notify) - return - } - dmap := make(map[string]string) - dmap["username"] = username - - configs, err := service.MyService.Storage().GetConfig() - if err != nil { - c.String(200, `

Failed to get rclone config:`+err.Error()+`

`) - notify["status"] = "fail" - notify["message"] = "Failed to get rclone config" - logger.Error("Then get config error: ", zap.Error(err), zap.Any("name", "dropbox")) - service.MyService.Notify().SendNotify("casaos:file:recover", notify) - return - } - for _, v := range configs.Remotes { - cf, err := service.MyService.Storage().GetConfigByName(v) - if err != nil { - logger.Error("then get config by name error: ", zap.Error(err), zap.Any("name", v)) - continue - } - if cf["type"] == "dropbox" && cf["username"] == dmap["username"] { - c.String(200, `

The same configuration has been added

`) - err := service.MyService.Storage().CheckAndMountByName(v) - if err != nil { - logger.Error("check and mount by name error: ", zap.Error(err), zap.Any("name", cf["username"])) - } - - notify["status"] = "warn" - notify["message"] = "The same configuration has been added" - service.MyService.Notify().SendNotify("casaos:file:recover", notify) - return - } - } - if len(username) > 0 { - a := strings.Split(username, "@") - username = a[0] - } - username += "_dropbox_" + strconv.FormatInt(time.Now().Unix(), 10) - - dmap["client_id"] = add.AppKey - dmap["client_secret"] = add.AppSecret - dmap["token"] = `{"access_token":"` + dropbox.AccessToken + `","token_type":"bearer","refresh_token":"` + dropbox.Addition.RefreshToken + `","expiry":"` + currentDate + `T` + currentTime.Add(time.Hour*3).Add(time.Minute*50).Format("15:04:05") + `.780385354Z"}` - dmap["mount_point"] = "/mnt/" + username - // data.SetValue(username, "type", "dropbox") - // data.SetValue(username, "client_id", add.AppKey) - // data.SetValue(username, "client_secret", add.AppSecret) - // data.SetValue(username, "mount_point", "/mnt/"+username) - - // data.SetValue(username, "token", `{"access_token":"`+dropbox.AccessToken+`","token_type":"bearer","refresh_token":"`+dropbox.Addition.RefreshToken+`","expiry":"`+currentDate+`T`+currentTime.Add(time.Hour*3).Format("15:04:05")+`.780385354Z"}`) - // e = data.Save() - // if e != nil { - // c.String(200, `

保存配置失败:`+e.Error()+`

`) - - // return - // } - service.MyService.Storage().CreateConfig(dmap, username, "dropbox") - service.MyService.Storage().MountStorage("/mnt/"+username, username+":") - - notify["status"] = "success" - notify["message"] = "Success" - service.MyService.Notify().SendNotify("casaos:file:recover", notify) - } - - c.String(200, `

Just close the page

`) -} diff --git a/route/v1/storage.go b/route/v1/storage.go deleted file mode 100644 index 0c07235..0000000 --- a/route/v1/storage.go +++ /dev/null @@ -1,143 +0,0 @@ -package v1 - -import ( - "strconv" - "strings" - - "github.com/IceWhaleTech/CasaOS-Common/utils/logger" - "github.com/IceWhaleTech/CasaOS/drivers/dropbox" - "github.com/IceWhaleTech/CasaOS/drivers/google_drive" - "github.com/IceWhaleTech/CasaOS/model" - "github.com/IceWhaleTech/CasaOS/pkg/utils/common_err" - "github.com/IceWhaleTech/CasaOS/pkg/utils/httper" - "github.com/IceWhaleTech/CasaOS/service" - "github.com/gin-gonic/gin" - "go.uber.org/zap" -) - -func ListStorages(c *gin.Context) { - // var req model.PageReq - // if err := c.ShouldBind(&req); err != nil { - // c.JSON(common_err.SUCCESS, model.Result{Success: common_err.CLIENT_ERROR, Message: common_err.GetMsg(common_err.CLIENT_ERROR), Data: err.Error()}) - // return - // } - // req.Validate() - - //logger.Info("ListStorages", zap.Any("req", req)) - //storages, total, err := service.MyService.Storage().GetStorages(req.Page, req.PerPage) - // if err != nil { - // c.JSON(common_err.SUCCESS, model.Result{Success: common_err.SERVICE_ERROR, Message: common_err.GetMsg(common_err.SERVICE_ERROR), Data: err.Error()}) - // return - // } - // c.JSON(common_err.SUCCESS, model.Result{Success: common_err.SUCCESS, Message: common_err.GetMsg(common_err.SUCCESS), Data: model.PageResp{ - // Content: storages, - // Total: total, - // }}) - r, err := service.MyService.Storage().GetStorages() - - if err != nil { - c.JSON(common_err.SUCCESS, model.Result{Success: common_err.SERVICE_ERROR, Message: common_err.GetMsg(common_err.SERVICE_ERROR), Data: err.Error()}) - return - } - - for i := 0; i < len(r.MountPoints); i++ { - dataMap, err := service.MyService.Storage().GetConfigByName(r.MountPoints[i].Fs) - if err != nil { - logger.Error("GetConfigByName", zap.Any("err", err)) - continue - } - if dataMap["type"] == "drive" { - r.MountPoints[i].Icon = google_drive.ICONURL - } - if dataMap["type"] == "dropbox" { - r.MountPoints[i].Icon = dropbox.ICONURL - } - r.MountPoints[i].Name = dataMap["username"] - } - list := []httper.MountPoint{} - - for _, v := range r.MountPoints { - list = append(list, httper.MountPoint{ - Fs: v.Fs, - Icon: v.Icon, - MountPoint: v.MountPoint, - Name: v.Name, - }) - } - - c.JSON(common_err.SUCCESS, model.Result{Success: common_err.SUCCESS, Message: common_err.GetMsg(common_err.SUCCESS), Data: list}) -} - -func UpdateStorage(c *gin.Context) { - var req model.Storage - if err := c.ShouldBind(&req); err != nil { - c.JSON(common_err.SUCCESS, model.Result{Success: common_err.CLIENT_ERROR, Message: common_err.GetMsg(common_err.CLIENT_ERROR), Data: err.Error()}) - return - } - if err := service.MyService.Storages().UpdateStorage(c, req); err != nil { - c.JSON(common_err.SUCCESS, model.Result{Success: common_err.SERVICE_ERROR, Message: common_err.GetMsg(common_err.SERVICE_ERROR), Data: err.Error()}) - } else { - c.JSON(common_err.SUCCESS, model.Result{Success: common_err.SUCCESS, Message: common_err.GetMsg(common_err.SUCCESS), Data: "success"}) - } -} - -func DeleteStorage(c *gin.Context) { - json := make(map[string]string) - c.ShouldBind(&json) - mountPoint := json["mount_point"] - if mountPoint == "" { - c.JSON(common_err.CLIENT_ERROR, model.Result{Success: common_err.CLIENT_ERROR, Message: common_err.GetMsg(common_err.CLIENT_ERROR), Data: "mount_point is empty"}) - return - } - err := service.MyService.Storage().UnmountStorage(mountPoint) - if err != nil { - c.JSON(common_err.SERVICE_ERROR, model.Result{Success: common_err.SERVICE_ERROR, Message: common_err.GetMsg(common_err.SERVICE_ERROR), Data: err.Error()}) - return - } - service.MyService.Storage().DeleteConfigByName(strings.ReplaceAll(mountPoint, "/mnt/", "")) - c.JSON(common_err.SUCCESS, model.Result{Success: common_err.SUCCESS, Message: common_err.GetMsg(common_err.SUCCESS), Data: "success"}) -} - -func DisableStorage(c *gin.Context) { - idStr := c.Query("id") - id, err := strconv.Atoi(idStr) - if err != nil { - c.JSON(common_err.SUCCESS, model.Result{Success: common_err.CLIENT_ERROR, Message: common_err.GetMsg(common_err.CLIENT_ERROR), Data: err.Error()}) - return - } - if err := service.MyService.Storages().DisableStorage(c, uint(id)); err != nil { - c.JSON(common_err.SUCCESS, model.Result{Success: common_err.SERVICE_ERROR, Message: common_err.GetMsg(common_err.SERVICE_ERROR), Data: err.Error()}) - return - } - c.JSON(common_err.SUCCESS, model.Result{Success: common_err.SUCCESS, Message: common_err.GetMsg(common_err.SUCCESS), Data: "success"}) -} - -func EnableStorage(c *gin.Context) { - idStr := c.Query("id") - id, err := strconv.Atoi(idStr) - if err != nil { - c.JSON(common_err.SUCCESS, model.Result{Success: common_err.CLIENT_ERROR, Message: common_err.GetMsg(common_err.CLIENT_ERROR), Data: err.Error()}) - return - } - if err := service.MyService.Storages().EnableStorage(c, uint(id)); err != nil { - c.JSON(common_err.SUCCESS, model.Result{Success: common_err.SERVICE_ERROR, Message: common_err.GetMsg(common_err.SERVICE_ERROR), Data: err.Error()}) - return - } - c.JSON(common_err.SUCCESS, model.Result{Success: common_err.SUCCESS, Message: common_err.GetMsg(common_err.SUCCESS), Data: "success"}) -} - -func GetStorage(c *gin.Context) { - - // idStr := c.Query("id") - // id, err := strconv.Atoi(idStr) - // if err != nil { - // c.JSON(common_err.SUCCESS, model.Result{Success: common_err.CLIENT_ERROR, Message: common_err.GetMsg(common_err.CLIENT_ERROR), Data: err.Error()}) - // return - // } - // storage, err := service.MyService.Storage().GetStorageById(uint(id)) - // if err != nil { - // c.JSON(common_err.SUCCESS, model.Result{Success: common_err.SERVICE_ERROR, Message: common_err.GetMsg(common_err.SERVICE_ERROR), Data: err.Error()}) - // return - // } - // c.JSON(common_err.SUCCESS, model.Result{Success: common_err.SUCCESS, Message: common_err.GetMsg(common_err.SUCCESS), Data: storage}) -} diff --git a/service/fs.go b/service/fs.go deleted file mode 100644 index d4dc218..0000000 --- a/service/fs.go +++ /dev/null @@ -1,154 +0,0 @@ -package service - -import ( - "context" - - "github.com/IceWhaleTech/CasaOS-Common/utils/logger" - "github.com/IceWhaleTech/CasaOS/internal/driver" - "github.com/IceWhaleTech/CasaOS/model" - log "github.com/dsoprea/go-logging" - "go.uber.org/zap" -) - -type FsService interface { - FList(ctx context.Context, path string, refresh ...bool) ([]model.Obj, error) - GetStorage(path string) (driver.Driver, error) - Link(ctx context.Context, path string, args model.LinkArgs) (*model.Link, model.Obj, error) -} - -type fsService struct { -} - -// the param named path of functions in this package is a mount path -// So, the purpose of this package is to convert mount path to actual path -// then pass the actual path to the op package - -func (f *fsService) FList(ctx context.Context, path string, refresh ...bool) ([]model.Obj, error) { - res, err := MyService.FsListService().FsList(ctx, path, refresh...) - if err != nil { - logger.Info("failed list", zap.Any("path", path), zap.Any("err", err)) - return nil, err - } - return res, nil -} - -// func (f *fsService) Get(ctx context.Context, path string) (model.Obj, error) { -// res, err := get(ctx, path) -// if err != nil { -// log.Errorf("failed get %s: %+v", path, err) -// return nil, err -// } -// return res, nil -// } - -func (f *fsService) Link(ctx context.Context, path string, args model.LinkArgs) (*model.Link, model.Obj, error) { - res, file, err := MyService.FsLinkService().Link(ctx, path, args) - if err != nil { - log.Errorf("failed link %s: %+v", path, err) - return nil, nil, err - } - return res, file, nil -} - -// func (f *fsService) MakeDir(ctx context.Context, path string, lazyCache ...bool) error { -// err := makeDir(ctx, path, lazyCache...) -// if err != nil { -// log.Errorf("failed make dir %s: %+v", path, err) -// } -// return err -// } - -// func (f *fsService) Move(ctx context.Context, srcPath, dstDirPath string, lazyCache ...bool) error { -// err := move(ctx, srcPath, dstDirPath, lazyCache...) -// if err != nil { -// log.Errorf("failed move %s to %s: %+v", srcPath, dstDirPath, err) -// } -// return err -// } - -// func (f *fsService) Copy(ctx context.Context, srcObjPath, dstDirPath string, lazyCache ...bool) (bool, error) { -// res, err := _copy(ctx, srcObjPath, dstDirPath, lazyCache...) -// if err != nil { -// log.Errorf("failed copy %s to %s: %+v", srcObjPath, dstDirPath, err) -// } -// return res, err -// } - -// func (f *fsService) Rename(ctx context.Context, srcPath, dstName string, lazyCache ...bool) error { -// err := rename(ctx, srcPath, dstName, lazyCache...) -// if err != nil { -// log.Errorf("failed rename %s to %s: %+v", srcPath, dstName, err) -// } -// return err -// } - -// func (f *fsService) Remove(ctx context.Context, path string) error { -// err := remove(ctx, path) -// if err != nil { -// log.Errorf("failed remove %s: %+v", path, err) -// } -// return err -// } - -// func PutDirectly(ctx context.Context, dstDirPath string, file *model.FileStream, lazyCache ...bool) error { -// err := putDirectly(ctx, dstDirPath, file, lazyCache...) -// if err != nil { -// log.Errorf("failed put %s: %+v", dstDirPath, err) -// } -// return err -// } - -// func (f *fsService) PutAsTask(dstDirPath string, file *model.FileStream) error { -// err := putAsTask(dstDirPath, file) -// if err != nil { -// log.Errorf("failed put %s: %+v", dstDirPath, err) -// } -// return err -// } - -func (f *fsService) GetStorage(path string) (driver.Driver, error) { - storageDriver, _, err := MyService.StoragePath().GetStorageAndActualPath(path) - if err != nil { - return nil, err - } - return storageDriver, nil -} - -// func (f *fsService) Other(ctx context.Context, args model.FsOtherArgs) (interface{}, error) { -// res, err := other(ctx, args) -// if err != nil { -// log.Errorf("failed remove %s: %+v", args.Path, err) -// } -// return res, err -// } - -// func get(ctx context.Context, path string) (model.Obj, error) { -// path = utils.FixAndCleanPath(path) -// // maybe a virtual file -// if path != "/" { -// virtualFiles := op.GetStorageVirtualFilesByPath(stdpath.Dir(path)) -// for _, f := range virtualFiles { -// if f.GetName() == stdpath.Base(path) { -// return f, nil -// } -// } -// } -// storage, actualPath, err := op.GetStorageAndActualPath(path) -// if err != nil { -// // if there are no storage prefix with path, maybe root folder -// if path == "/" { -// return &model.Object{ -// Name: "root", -// Size: 0, -// Modified: time.Time{}, -// IsFolder: true, -// }, nil -// } -// return nil, errors.WithMessage(err, "failed get storage") -// } -// return op.Get(ctx, storage, actualPath) -// } - -func NewFsService() FsService { - return &fsService{} -} diff --git a/service/fs_link.go b/service/fs_link.go deleted file mode 100644 index bf4c013..0000000 --- a/service/fs_link.go +++ /dev/null @@ -1,27 +0,0 @@ -package service - -import ( - "context" - - "github.com/IceWhaleTech/CasaOS/internal/op" - "github.com/IceWhaleTech/CasaOS/model" - "github.com/pkg/errors" -) - -type FsLinkService interface { - Link(ctx context.Context, path string, args model.LinkArgs) (*model.Link, model.Obj, error) -} - -type fsLinkService struct { -} - -func (f *fsLinkService) Link(ctx context.Context, path string, args model.LinkArgs) (*model.Link, model.Obj, error) { - storage, actualPath, err := MyService.StoragePath().GetStorageAndActualPath(path) - if err != nil { - return nil, nil, errors.WithMessage(err, "failed get storage") - } - return op.Link(ctx, storage, actualPath, args) -} -func NewFsLinkService() FsLinkService { - return &fsLinkService{} -} diff --git a/service/fs_list.go b/service/fs_list.go deleted file mode 100644 index 24219b0..0000000 --- a/service/fs_list.go +++ /dev/null @@ -1,198 +0,0 @@ -package service - -import ( - "context" - stdpath "path" - "time" - - "github.com/IceWhaleTech/CasaOS-Common/utils/logger" - "github.com/IceWhaleTech/CasaOS/internal/driver" - "github.com/IceWhaleTech/CasaOS/internal/op" - "github.com/IceWhaleTech/CasaOS/model" - "github.com/IceWhaleTech/CasaOS/pkg/singleflight" - "github.com/IceWhaleTech/CasaOS/pkg/utils" - "github.com/Xhofe/go-cache" - - log "github.com/dsoprea/go-logging" - "github.com/pkg/errors" - "go.uber.org/zap" -) - -type FsListService interface { - FsList(ctx context.Context, path string, refresh ...bool) ([]model.Obj, error) - Key(storage driver.Driver, path string) string - Get(ctx context.Context, storage driver.Driver, path string) (model.Obj, error) - GetUnwrap(ctx context.Context, storage driver.Driver, path string) (model.Obj, error) - List(ctx context.Context, storage driver.Driver, path string, args model.ListArgs, refresh ...bool) ([]model.Obj, error) -} - -type fsListService struct { -} - -var listCache = cache.NewMemCache(cache.WithShards[[]model.Obj](64)) -var listG singleflight.Group[[]model.Obj] - -// List files -func (fl *fsListService) FsList(ctx context.Context, path string, refresh ...bool) ([]model.Obj, error) { - - virtualFiles := MyService.Storages().GetStorageVirtualFilesByPath(path) - storage, actualPath, err := MyService.StoragePath().GetStorageAndActualPath(path) - if err != nil && len(virtualFiles) == 0 { - return nil, errors.WithMessage(err, "failed get storage") - } - - var _objs []model.Obj - if storage != nil { - _objs, err = fl.List(ctx, storage, actualPath, model.ListArgs{ - ReqPath: path, - }, refresh...) - if err != nil { - log.Errorf("%+v", err) - if len(virtualFiles) == 0 { - return nil, errors.WithMessage(err, "failed get objs") - } - } - } - - om := model.NewObjMerge() - - objs := om.Merge(virtualFiles, _objs...) - return objs, nil -} - -func (fl *fsListService) Key(storage driver.Driver, path string) string { - return stdpath.Join(storage.GetStorage().MountPath, utils.FixAndCleanPath(path)) -} - -// Get object from list of files -func (fl *fsListService) Get(ctx context.Context, storage driver.Driver, path string) (model.Obj, error) { - path = utils.FixAndCleanPath(path) - logger.Info("get", zap.String("path", path)) - - // is root folder - if utils.PathEqual(path, "/") { - var rootObj model.Obj - switch r := storage.GetAddition().(type) { - case driver.IRootId: - rootObj = &model.Object{ - ID: r.GetRootId(), - Name: op.RootName, - Size: 0, - Modified: storage.GetStorage().Modified, - IsFolder: true, - } - case driver.IRootPath: - rootObj = &model.Object{ - Path: r.GetRootPath(), - Name: op.RootName, - Size: 0, - Modified: storage.GetStorage().Modified, - IsFolder: true, - } - default: - if storage, ok := storage.(driver.Getter); ok { - obj, err := storage.GetRoot(ctx) - if err != nil { - return nil, errors.WithMessage(err, "failed get root obj") - } - rootObj = obj - } - } - if rootObj == nil { - return nil, errors.Errorf("please implement IRootPath or IRootId or Getter method") - } - return &model.ObjWrapName{ - Name: op.RootName, - Obj: rootObj, - }, nil - } - - // not root folder - dir, name := stdpath.Split(path) - files, err := fl.List(ctx, storage, dir, model.ListArgs{}) - if err != nil { - return nil, errors.WithMessage(err, "failed get parent list") - } - for _, f := range files { - // TODO maybe copy obj here - if f.GetName() == name { - return f, nil - } - } - logger.Info("cant find obj with name", zap.Any("name", name)) - return nil, errors.WithStack(errors.New("object not found")) -} - -func (fl *fsListService) GetUnwrap(ctx context.Context, storage driver.Driver, path string) (model.Obj, error) { - obj, err := fl.Get(ctx, storage, path) - if err != nil { - return nil, err - } - return model.UnwrapObjs(obj), err -} - -// List files in storage, not contains virtual file -func (fl *fsListService) List(ctx context.Context, storage driver.Driver, path string, args model.ListArgs, refresh ...bool) ([]model.Obj, error) { - if storage.Config().CheckStatus && storage.GetStorage().Status != op.WORK { - return nil, errors.Errorf("storage not init: %s", storage.GetStorage().Status) - } - path = utils.FixAndCleanPath(path) - logger.Info("op.List", zap.Any("path", path)) - key := fl.Key(storage, path) - if !utils.IsBool(refresh...) { - if files, ok := listCache.Get(key); ok { - logger.Info("op.List", zap.Any("use cache", path)) - return files, nil - } - } - dir, err := fl.GetUnwrap(ctx, storage, path) - if err != nil { - return nil, errors.WithMessage(err, "failed get dir") - } - logger.Info("op.List", zap.Any("dir", dir)) - if !dir.IsDir() { - return nil, errors.WithStack(errors.New("not a folder")) - } - objs, err, _ := listG.Do(key, func() ([]model.Obj, error) { - files, err := storage.List(ctx, dir, args) - if err != nil { - return nil, errors.Wrapf(err, "failed to list objs") - } - // set path - for _, f := range files { - if s, ok := f.(model.SetPath); ok && f.GetPath() == "" && dir.GetPath() != "" { - s.SetPath(stdpath.Join(dir.GetPath(), f.GetName())) - } - } - // warp obj name - model.WrapObjsName(files) - // call hooks - go func(reqPath string, files []model.Obj) { - for _, hook := range op.ObjsUpdateHooks { - hook(args.ReqPath, files) - } - }(args.ReqPath, files) - - // sort objs - if storage.Config().LocalSort { - model.SortFiles(files, storage.GetStorage().OrderBy, storage.GetStorage().OrderDirection) - } - model.ExtractFolder(files, storage.GetStorage().ExtractFolder) - - if !storage.Config().NoCache { - if len(files) > 0 { - logger.Info("set cache", zap.Any("key", key), zap.Any("files", files)) - listCache.Set(key, files, cache.WithEx[[]model.Obj](time.Minute*time.Duration(storage.GetStorage().CacheExpiration))) - } else { - logger.Info("del cache", zap.Any("key", key)) - listCache.Del(key) - } - } - return files, nil - }) - return objs, err -} - -func NewFsListService() FsListService { - return &fsListService{} -} diff --git a/service/service.go b/service/service.go index 4c2f96b..4bdec77 100644 --- a/service/service.go +++ b/service/service.go @@ -39,12 +39,7 @@ type Repository interface { Rely() RelyService Shares() SharesService System() SystemService - Storage() StorageService - Storages() StoragesService - StoragePath() StoragePathService - FsListService() FsListService - FsLinkService() FsLinkService - FsService() FsService + MessageBus() *message_bus.ClientWithResponses Other() OtherService } @@ -56,70 +51,37 @@ func NewService(db *gorm.DB, RuntimePath string) Repository { } return &store{ - casa: NewCasaService(), - connections: NewConnectionsService(db), - gateway: gatewayManagement, - notify: NewNotifyService(db), - rely: NewRelyService(db), - system: NewSystemService(), - health: NewHealthService(), - shares: NewSharesService(db), - storage: NewStorageService(), - storages: NewStoragesService(), - storage_path: NewStoragePathService(), - fs_list: NewFsListService(), - fs_link: NewFsLinkService(), - fs: NewFsService(), - other: NewOtherService(), + casa: NewCasaService(), + connections: NewConnectionsService(db), + gateway: gatewayManagement, + notify: NewNotifyService(db), + rely: NewRelyService(db), + system: NewSystemService(), + health: NewHealthService(), + shares: NewSharesService(db), + + other: NewOtherService(), } } type store struct { - db *gorm.DB - casa CasaService - notify NotifyServer - rely RelyService - system SystemService - shares SharesService - connections ConnectionsService - gateway external.ManagementService - storage StorageService - storages StoragesService - storage_path StoragePathService - fs_list FsListService - fs_link FsLinkService - fs FsService - health HealthService - other OtherService + db *gorm.DB + casa CasaService + notify NotifyServer + rely RelyService + system SystemService + shares SharesService + connections ConnectionsService + gateway external.ManagementService + + health HealthService + other OtherService } func (c *store) Other() OtherService { return c.other } -func (c *store) FsLinkService() FsLinkService { - return c.fs_link -} -func (c *store) FsService() FsService { - return c.fs -} - -func (c *store) FsListService() FsListService { - return c.fs_list -} - -func (c *store) StoragePath() StoragePathService { - return c.storage_path -} - -func (c *store) Storages() StoragesService { - return c.storages -} - -func (c *store) Storage() StorageService { - return c.storage -} - func (c *store) Gateway() external.ManagementService { return c.gateway } diff --git a/service/storage.go b/service/storage.go deleted file mode 100644 index a13711e..0000000 --- a/service/storage.go +++ /dev/null @@ -1,116 +0,0 @@ -package service - -import ( - "io/ioutil" - - "github.com/IceWhaleTech/CasaOS-Common/utils/logger" - "github.com/IceWhaleTech/CasaOS/pkg/utils/file" - "github.com/IceWhaleTech/CasaOS/pkg/utils/httper" - "go.uber.org/zap" -) - -type StorageService interface { - MountStorage(mountPoint, fs string) error - UnmountStorage(mountPoint string) error - GetStorages() (httper.MountList, error) - CreateConfig(data map[string]string, name string, t string) error - CheckAndMountByName(name string) error - CheckAndMountAll() error - GetConfigByName(name string) (map[string]string, error) - DeleteConfigByName(name string) error - GetConfig() (httper.RemotesResult, error) -} - -type storageStruct struct { -} - -func (s *storageStruct) MountStorage(mountPoint, fs string) error { - file.IsNotExistMkDir(mountPoint) - return httper.Mount(mountPoint, fs) -} -func (s *storageStruct) UnmountStorage(mountPoint string) error { - err := httper.Unmount(mountPoint) - if err == nil { - dir, _ := ioutil.ReadDir(mountPoint) - - if len(dir) == 0 { - file.RMDir(mountPoint) - } - return nil - } - return err -} -func (s *storageStruct) GetStorages() (httper.MountList, error) { - return httper.GetMountList() -} -func (s *storageStruct) CreateConfig(data map[string]string, name string, t string) error { - httper.CreateConfig(data, name, t) - return nil -} -func (s *storageStruct) CheckAndMountByName(name string) error { - storages, _ := MyService.Storage().GetStorages() - currentRemote, _ := httper.GetConfigByName(name) - mountPoint := currentRemote["mount_point"] - isMount := false - for _, v := range storages.MountPoints { - if v.MountPoint == mountPoint { - isMount = true - break - } - } - if !isMount { - return MyService.Storage().MountStorage(mountPoint, name+":") - } - return nil -} -func (s *storageStruct) CheckAndMountAll() error { - storages, err := MyService.Storage().GetStorages() - if err != nil { - return err - } - logger.Info("when CheckAndMountAll storages", zap.Any("storages", storages)) - section, err := httper.GetAllConfigName() - if err != nil { - return err - } - logger.Info("when CheckAndMountAll section", zap.Any("section", section)) - for _, v := range section.Remotes { - currentRemote, _ := httper.GetConfigByName(v) - mountPoint := currentRemote["mount_point"] - if len(mountPoint) == 0 { - continue - } - isMount := false - for _, v := range storages.MountPoints { - if v.MountPoint == mountPoint { - isMount = true - break - } - } - if !isMount { - logger.Info("when CheckAndMountAll MountStorage", zap.String("mountPoint", mountPoint), zap.String("fs", v)) - err := MyService.Storage().MountStorage(mountPoint, v+":") - if err != nil { - logger.Error("when CheckAndMountAll then", zap.String("mountPoint", mountPoint), zap.String("fs", v), zap.Error(err)) - } - } - } - return nil -} - -func (s *storageStruct) GetConfigByName(name string) (map[string]string, error) { - return httper.GetConfigByName(name) -} -func (s *storageStruct) DeleteConfigByName(name string) error { - return httper.DeleteConfigByName(name) -} -func (s *storageStruct) GetConfig() (httper.RemotesResult, error) { - section, err := httper.GetAllConfigName() - if err != nil { - return httper.RemotesResult{}, err - } - return section, nil -} -func NewStorageService() StorageService { - return &storageStruct{} -} diff --git a/service/storage_path.go b/service/storage_path.go deleted file mode 100644 index 5d93268..0000000 --- a/service/storage_path.go +++ /dev/null @@ -1,34 +0,0 @@ -package service - -import ( - "strings" - - "github.com/IceWhaleTech/CasaOS-Common/utils/logger" - "github.com/IceWhaleTech/CasaOS/internal/driver" - "github.com/IceWhaleTech/CasaOS/pkg/utils" - "github.com/pkg/errors" - "go.uber.org/zap" -) - -type StoragePathService interface { - GetStorageAndActualPath(rawPath string) (storage driver.Driver, actualPath string, err error) -} - -type storagePathStruct struct { -} - -func (s *storagePathStruct) GetStorageAndActualPath(rawPath string) (storage driver.Driver, actualPath string, err error) { - rawPath = utils.FixAndCleanPath(rawPath) - storage = MyService.Storages().GetBalancedStorage(rawPath) - if storage == nil { - err = errors.Errorf("can't find storage with rawPath: %s", rawPath) - return - } - logger.Info("use storage", zap.Any("storage mount path", storage.GetStorage().MountPath)) - mountPath := utils.GetActualMountPath(storage.GetStorage().MountPath) - actualPath = utils.FixAndCleanPath(strings.TrimPrefix(rawPath, mountPath)) - return -} -func NewStoragePathService() StoragePathService { - return &storagePathStruct{} -} diff --git a/service/storage_service.go b/service/storage_service.go deleted file mode 100644 index 30cfeed..0000000 --- a/service/storage_service.go +++ /dev/null @@ -1,398 +0,0 @@ -package service - -import ( - "context" - "sort" - "strings" - "time" - - "github.com/IceWhaleTech/CasaOS-Common/utils/logger" - "github.com/IceWhaleTech/CasaOS/pkg/utils" - jsoniter "github.com/json-iterator/go" - "go.uber.org/zap" - - "github.com/IceWhaleTech/CasaOS/pkg/generic_sync" - - "github.com/IceWhaleTech/CasaOS/model" - - "github.com/IceWhaleTech/CasaOS/internal/driver" - "github.com/IceWhaleTech/CasaOS/internal/op" - mapset "github.com/deckarep/golang-set/v2" - "github.com/pkg/errors" -) - -type StoragesService interface { - HasStorage(mountPath string) bool - CreateStorage(ctx context.Context, storage model.Storage) (uint, error) - LoadStorage(ctx context.Context, storage model.Storage) error - EnableStorage(ctx context.Context, id uint) error - DisableStorage(ctx context.Context, id uint) error - UpdateStorage(ctx context.Context, storage model.Storage) error - DeleteStorageById(ctx context.Context, id uint) error - MustSaveDriverStorage(driver driver.Driver) error - GetStorageVirtualFilesByPath(prefix string) []model.Obj - initStorage(ctx context.Context, storage model.Storage, storageDriver driver.Driver, setMountPath func(d driver.Driver, ctx context.Context) string) (err error) - InitStorages() - GetBalancedStorage(path string) driver.Driver -} - -type storagesStruct struct { -} - -// Although the driver type is stored, -// there is a storage in each driver, -// so it should actually be a storage, just wrapped by the driver -var storagesMap generic_sync.MapOf[string, driver.Driver] - -func GetAllStorages() []driver.Driver { - return storagesMap.Values() -} - -func (s *storagesStruct) HasStorage(mountPath string) bool { - return storagesMap.Has(utils.FixAndCleanPath(mountPath)) -} - -func GetStorageByMountPath(mountPath string) (driver.Driver, error) { - mountPath = utils.FixAndCleanPath(mountPath) - storageDriver, ok := storagesMap.Load(mountPath) - if !ok { - return nil, errors.Errorf("no mount path for an storage is: %s", mountPath) - } - return storageDriver, nil -} - -// CreateStorage Save the storage to database so storage can get an id -// then instantiate corresponding driver and save it in memory -func (s *storagesStruct) CreateStorage(ctx context.Context, storage model.Storage) (uint, error) { - storage.Modified = time.Now() - storage.MountPath = utils.FixAndCleanPath(storage.MountPath) - var err error - // check driver first - driverName := storage.Driver - driverNew, err := op.GetDriverNew(driverName) - if err != nil { - return 0, errors.WithMessage(err, "failed get driver new") - } - storageDriver := driverNew() - // // insert storage to database - // err = MyService.Storage().CreateStorage(&storage) - // if err != nil { - - // return storage.ID, errors.WithMessage(err, "failed create storage in database") - // } - // already has an id - err = s.initStorage(ctx, storage, storageDriver, func(d driver.Driver, ctx context.Context) string { - u, _ := d.GetUserInfo(ctx) - if len(u) > 0 { - a := strings.Split(u, "@") - u = a[0] - } - return u - }) - if err != nil { - s.DeleteStorageById(ctx, storage.ID) - return storage.ID, errors.Wrap(err, "failed init storage") - } - - go op.CallStorageHooks("add", storageDriver) - - logger.Error("storage created", zap.Any("storage", storageDriver)) - return storage.ID, nil -} - -// LoadStorage load exist storage in db to memory -func (s *storagesStruct) LoadStorage(ctx context.Context, storage model.Storage) error { - storage.MountPath = utils.FixAndCleanPath(storage.MountPath) - // check driver first - driverName := storage.Driver - driverNew, err := op.GetDriverNew(driverName) - if err != nil { - return errors.WithMessage(err, "failed get driver new") - } - storageDriver := driverNew() - - err = s.initStorage(ctx, storage, storageDriver, nil) - go op.CallStorageHooks("add", storageDriver) - logger.Info("storage created", zap.Any("storage", storageDriver)) - return err -} - -// initStorage initialize the driver and store to storagesMap -func (s *storagesStruct) initStorage(ctx context.Context, storage model.Storage, storageDriver driver.Driver, setMountPath func(d driver.Driver, ctx context.Context) string) (err error) { - storageDriver.SetStorage(storage) - driverStorage := storageDriver.GetStorage() - - // Unmarshal Addition - - var json = jsoniter.ConfigCompatibleWithStandardLibrary - - err = json.UnmarshalFromString(driverStorage.Addition, storageDriver.GetAddition()) - if err == nil { - err = storageDriver.Init(ctx) - } - if setMountPath != nil { - driverStorage.MountPath += "_" + setMountPath(storageDriver, ctx) - - } - if s.HasStorage(driverStorage.MountPath) { - return errors.New("mount path already exists") - } - storageDriver.SetStorage(*driverStorage) - storagesMap.Store(driverStorage.MountPath, storageDriver) - - if err != nil { - driverStorage.SetStatus(err.Error()) - err = errors.Wrap(err, "failed init storage") - } else { - driverStorage.SetStatus(op.WORK) - } - - err = s.MustSaveDriverStorage(storageDriver) - - return err -} - -func (s *storagesStruct) EnableStorage(ctx context.Context, id uint) error { - // storage, err := MyService.Storage().GetStorageById(id) - // if err != nil { - // return errors.WithMessage(err, "failed get storage") - // } - // if !storage.Disabled { - // return errors.Errorf("this storage have enabled") - // } - // storage.Disabled = false - // err = MyService.Storage().UpdateStorage(storage) - // if err != nil { - // return errors.WithMessage(err, "failed update storage in db") - // } - // err = s.LoadStorage(ctx, *storage) - // if err != nil { - // return errors.WithMessage(err, "failed load storage") - // } - return nil -} - -func (s *storagesStruct) DisableStorage(ctx context.Context, id uint) error { - // storage, err := MyService.Storage().GetStorageById(id) - // if err != nil { - // return errors.WithMessage(err, "failed get storage") - // } - // if storage.Disabled { - // return errors.Errorf("this storage have disabled") - // } - // storageDriver, err := GetStorageByMountPath(storage.MountPath) - // if err != nil { - // return errors.WithMessage(err, "failed get storage driver") - // } - // // drop the storage in the driver - // if err := storageDriver.Drop(ctx); err != nil { - // return errors.Wrap(err, "failed drop storage") - // } - // // delete the storage in the memory - // storage.Disabled = true - // err = MyService.Storage().UpdateStorage(storage) - // if err != nil { - // return errors.WithMessage(err, "failed update storage in db") - // } - // storagesMap.Delete(storage.MountPath) - // go op.CallStorageHooks("del", storageDriver) - return nil -} - -// UpdateStorage update storage -// get old storage first -// drop the storage then reinitialize -func (s *storagesStruct) UpdateStorage(ctx context.Context, storage model.Storage) error { - // oldStorage, err := MyService.Storage().GetStorageById(storage.ID) - // if err != nil { - // return errors.WithMessage(err, "failed get old storage") - // } - // if oldStorage.Driver != storage.Driver { - // return errors.Errorf("driver cannot be changed") - // } - // storage.Modified = time.Now() - // storage.MountPath = utils.FixAndCleanPath(storage.MountPath) - // err = MyService.Storage().UpdateStorage(&storage) - // if err != nil { - // return errors.WithMessage(err, "failed update storage in database") - // } - // if storage.Disabled { - // return nil - // } - // storageDriver, err := GetStorageByMountPath(oldStorage.MountPath) - // if oldStorage.MountPath != storage.MountPath { - // // mount path renamed, need to drop the storage - // storagesMap.Delete(oldStorage.MountPath) - // } - // if err != nil { - // return errors.WithMessage(err, "failed get storage driver") - // } - // err = storageDriver.Drop(ctx) - // if err != nil { - // return errors.Wrapf(err, "failed drop storage") - // } - - // err = s.initStorage(ctx, storage, storageDriver, nil) - // go op.CallStorageHooks("update", storageDriver) - - // logger.Info("storage updated", zap.Any("storage", storageDriver)) - //return err - return nil -} - -func (s *storagesStruct) DeleteStorageById(ctx context.Context, id uint) error { - // storage, err := MyService.Storage().GetStorageById(id) - // if err != nil { - // return errors.WithMessage(err, "failed get storage") - // } - // if !storage.Disabled { - // storageDriver, err := GetStorageByMountPath(storage.MountPath) - // if err == nil { - // // drop the storage in the driver - // if err := storageDriver.Drop(ctx); err != nil { - // return errors.Wrapf(err, "failed drop storage") - // } - // // delete the storage in the memory - // storagesMap.Delete(storage.MountPath) - // } - - // go op.CallStorageHooks("del", storageDriver) - // } - // // delete the storage in the database - // if err := MyService.Storage().DeleteStorageById(id); err != nil { - // return errors.WithMessage(err, "failed delete storage in database") - // } - return nil -} - -// MustSaveDriverStorage call from specific driver -func (s *storagesStruct) MustSaveDriverStorage(driver driver.Driver) error { - err := saveDriverStorage(driver) - if err != nil { - logger.Error("failed save driver storage", zap.Any("err", err)) - } - return err -} - -func saveDriverStorage(driver driver.Driver) error { - // storage := driver.GetStorage() - // addition := driver.GetAddition() - - // var json = jsoniter.ConfigCompatibleWithStandardLibrary - - // str, err := json.MarshalToString(addition) - // if err != nil { - // return errors.Wrap(err, "error while marshal addition") - // } - // storage.Addition = str - // err = MyService.Storage().UpdateStorage(storage) - // if err != nil { - // return errors.WithMessage(err, "failed update storage in database") - // } - return nil -} - -// getStoragesByPath get storage by longest match path, contains balance storage. -// for example, there is /a/b,/a/c,/a/d/e,/a/d/e.balance -// getStoragesByPath(/a/d/e/f) => /a/d/e,/a/d/e.balance -func getStoragesByPath(path string) []driver.Driver { - storages := make([]driver.Driver, 0) - curSlashCount := 0 - storagesMap.Range(func(mountPath string, value driver.Driver) bool { - mountPath = utils.GetActualMountPath(mountPath) - // is this path - if utils.IsSubPath(mountPath, path) { - slashCount := strings.Count(utils.PathAddSeparatorSuffix(mountPath), "/") - // not the longest match - if slashCount > curSlashCount { - storages = storages[:0] - curSlashCount = slashCount - } - if slashCount == curSlashCount { - storages = append(storages, value) - } - } - return true - }) - // make sure the order is the same for same input - sort.Slice(storages, func(i, j int) bool { - return storages[i].GetStorage().MountPath < storages[j].GetStorage().MountPath - }) - return storages -} - -// GetStorageVirtualFilesByPath Obtain the virtual file generated by the storage according to the path -// for example, there are: /a/b,/a/c,/a/d/e,/a/b.balance1,/av -// GetStorageVirtualFilesByPath(/a) => b,c,d -func (s *storagesStruct) GetStorageVirtualFilesByPath(prefix string) []model.Obj { - files := make([]model.Obj, 0) - storages := storagesMap.Values() - sort.Slice(storages, func(i, j int) bool { - if storages[i].GetStorage().Order == storages[j].GetStorage().Order { - return storages[i].GetStorage().MountPath < storages[j].GetStorage().MountPath - } - return storages[i].GetStorage().Order < storages[j].GetStorage().Order - }) - - prefix = utils.FixAndCleanPath(prefix) - set := mapset.NewSet[string]() - for _, v := range storages { - mountPath := utils.GetActualMountPath(v.GetStorage().MountPath) - // Exclude prefix itself and non prefix - if len(prefix) >= len(mountPath) || !utils.IsSubPath(prefix, mountPath) { - continue - } - name := strings.SplitN(strings.TrimPrefix(mountPath[len(prefix):], "/"), "/", 2)[0] - if set.Add(name) { - files = append(files, &model.Object{ - Name: name, - Size: 0, - Modified: v.GetStorage().Modified, - IsFolder: true, - }) - } - } - return files -} - -var balanceMap generic_sync.MapOf[string, int] - -// GetBalancedStorage get storage by path -func (s *storagesStruct) GetBalancedStorage(path string) driver.Driver { - path = utils.FixAndCleanPath(path) - storages := getStoragesByPath(path) - storageNum := len(storages) - switch storageNum { - case 0: - return nil - case 1: - return storages[0] - default: - virtualPath := utils.GetActualMountPath(storages[0].GetStorage().MountPath) - i, _ := balanceMap.LoadOrStore(virtualPath, 0) - i = (i + 1) % storageNum - balanceMap.Store(virtualPath, i) - return storages[i] - } -} -func (s *storagesStruct) InitStorages() { - // storages, err := MyService.Storage().GetEnabledStorages() - // if err != nil { - // logger.Error("failed get enabled storages", zap.Any("err", err)) - // } - // go func(storages []model.Storage) { - // for i := range storages { - // err := s.LoadStorage(context.Background(), storages[i]) - // if err != nil { - // logger.Error("failed get enabled storages", zap.Any("err", err)) - // } else { - // logger.Info("success load storage", zap.String("mount_path", storages[i].MountPath)) - // } - // } - // conf.StoragesLoaded = true - // }(storages) - -} -func NewStoragesService() StoragesService { - return &storagesStruct{} -}