casdoor/object/storage.go

174 lines
4.8 KiB
Go
Raw Normal View History

2022-02-13 23:39:27 +08:00
// Copyright 2021 The Casdoor Authors. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package object
import (
"bytes"
"fmt"
2022-07-12 23:24:24 +08:00
"net/url"
"path/filepath"
"strings"
"github.com/casdoor/casdoor/conf"
"github.com/casdoor/casdoor/i18n"
2022-01-20 14:11:46 +08:00
"github.com/casdoor/casdoor/storage"
"github.com/casdoor/casdoor/util"
"github.com/casdoor/oss"
)
2021-12-04 16:35:33 +08:00
var isCloudIntranet bool
func init() {
isCloudIntranet = conf.GetConfigBool("isCloudIntranet")
2021-12-04 16:35:33 +08:00
}
func getProviderEndpoint(provider *Provider) string {
endpoint := provider.Endpoint
if provider.IntranetEndpoint != "" && isCloudIntranet {
endpoint = provider.IntranetEndpoint
}
return endpoint
}
2022-07-12 23:24:24 +08:00
func escapePath(path string) string {
tokens := strings.Split(path, "/")
if len(tokens) > 0 {
tokens[len(tokens)-1] = url.QueryEscape(tokens[len(tokens)-1])
}
res := strings.Join(tokens, "/")
return res
}
func GetTruncatedPath(provider *Provider, fullFilePath string, limit int) string {
pathPrefix := util.UrlJoin(util.GetUrlPath(provider.Domain), provider.PathPrefix)
dir, file := filepath.Split(fullFilePath)
ext := filepath.Ext(file)
fileName := strings.TrimSuffix(file, ext)
for {
escapedString := escapePath(escapePath(fullFilePath))
if len(escapedString) < limit-len(pathPrefix) {
break
}
rs := []rune(fileName)
fileName = string(rs[0 : len(rs)-1])
fullFilePath = dir + fileName + ext
}
return fullFilePath
}
func GetUploadFileUrl(provider *Provider, fullFilePath string, hasTimestamp bool) (string, string) {
escapedPath := util.UrlJoin(provider.PathPrefix, fullFilePath)
2022-07-12 23:24:24 +08:00
objectKey := util.UrlJoin(util.GetUrlPath(provider.Domain), escapedPath)
2021-08-21 23:17:33 +08:00
host := ""
if provider.Type != "Local File System" {
// provider.Domain = "https://cdn.casbin.com/casdoor/"
host = util.GetUrlHost(provider.Domain)
if !strings.HasPrefix(host, "http://") && !strings.HasPrefix(host, "https://") {
host = fmt.Sprintf("https://%s", host)
}
} else {
2022-02-13 20:47:34 +08:00
// provider.Domain = "http://localhost:8000" or "https://door.casdoor.com"
2021-08-21 23:17:33 +08:00
host = util.UrlJoin(provider.Domain, "/files")
}
if provider.Type == "Azure Blob" {
host = util.UrlJoin(host, provider.Bucket)
}
fileUrl := util.UrlJoin(host, escapePath(objectKey))
2021-08-21 23:17:33 +08:00
if hasTimestamp {
fileUrl = fmt.Sprintf("%s?t=%s", fileUrl, util.GetCurrentUnixTime())
2021-08-21 23:17:33 +08:00
}
if provider.Type == "Tencent Cloud COS" {
objectKey = escapePath(objectKey)
}
2021-08-21 23:17:33 +08:00
return fileUrl, objectKey
}
func getStorageProvider(provider *Provider, lang string) (oss.StorageInterface, error) {
2021-12-04 16:35:33 +08:00
endpoint := getProviderEndpoint(provider)
storageProvider := storage.GetStorageProvider(provider.Type, provider.ClientId, provider.ClientSecret, provider.RegionId, provider.Bucket, endpoint)
if storageProvider == nil {
return nil, fmt.Errorf(i18n.Translate(lang, "storage:The provider type: %s is not supported"), provider.Type)
}
if provider.Domain == "" {
provider.Domain = storageProvider.GetEndpoint()
UpdateProvider(provider.GetId(), provider)
}
return storageProvider, nil
}
func uploadFile(provider *Provider, fullFilePath string, fileBuffer *bytes.Buffer, lang string) (string, string, error) {
storageProvider, err := getStorageProvider(provider, lang)
if err != nil {
return "", "", err
}
fileUrl, objectKey := GetUploadFileUrl(provider, fullFilePath, true)
2021-08-21 23:17:33 +08:00
_, err = storageProvider.Put(objectKey, fileBuffer)
if err != nil {
2021-08-15 00:17:53 +08:00
return "", "", err
}
2021-08-15 00:17:53 +08:00
return fileUrl, objectKey, nil
}
2021-08-15 00:41:51 +08:00
2023-03-19 21:47:49 +08:00
func UploadFileSafe(provider *Provider, fullFilePath string, fileBuffer *bytes.Buffer, lang string) (string, string, error) {
// check fullFilePath is there security issue
if strings.Contains(fullFilePath, "..") {
return "", "", fmt.Errorf("the fullFilePath: %s is not allowed", fullFilePath)
}
2021-11-20 16:21:15 +08:00
var fileUrl string
var objectKey string
var err error
times := 0
for {
2023-03-19 21:47:49 +08:00
fileUrl, objectKey, err = uploadFile(provider, fullFilePath, fileBuffer, lang)
2021-11-20 16:21:15 +08:00
if err != nil {
times += 1
if times >= 5 {
return "", "", err
}
} else {
break
}
}
return fileUrl, objectKey, nil
}
func DeleteFile(provider *Provider, objectKey string, lang string) error {
// check fullFilePath is there security issue
if strings.Contains(objectKey, "..") {
return fmt.Errorf(i18n.Translate(lang, "storage:The objectKey: %s is not allowed"), objectKey)
}
storageProvider, err := getStorageProvider(provider, lang)
if err != nil {
return err
2021-08-15 00:41:51 +08:00
}
return storageProvider.Delete(objectKey)
}