From 7e4246236f2fd0df8e2de5545f89fff798aa37c1 Mon Sep 17 00:00:00 2001 From: Giuseppe Lo Presti Date: Tue, 23 Apr 2024 09:15:11 +0200 Subject: [PATCH] WIP to implement shares caching into the gateway --- internal/grpc/services/gateway/gateway.go | 78 +++++++++++++++---- .../services/gateway/usershareprovider.go | 1 + .../handlers/apps/sharing/shares/shares.go | 48 ++++-------- 3 files changed, 77 insertions(+), 50 deletions(-) diff --git a/internal/grpc/services/gateway/gateway.go b/internal/grpc/services/gateway/gateway.go index d18ad85021..f750335f58 100644 --- a/internal/grpc/services/gateway/gateway.go +++ b/internal/grpc/services/gateway/gateway.go @@ -29,10 +29,12 @@ import ( gateway "github.com/cs3org/go-cs3apis/cs3/gateway/v1beta1" "github.com/cs3org/reva/pkg/errtypes" "github.com/cs3org/reva/pkg/rgrpc" + "github.com/cs3org/reva/pkg/share/cache" "github.com/cs3org/reva/pkg/sharedconf" "github.com/cs3org/reva/pkg/token" "github.com/cs3org/reva/pkg/token/manager/registry" "github.com/cs3org/reva/pkg/utils/cfg" + "github.com/cs3org/reva/pkg/utils/resourceid" "google.golang.org/grpc" ) @@ -65,13 +67,18 @@ type config struct { TransferExpires int64 `mapstructure:"transfer_expires"` TokenManager string `mapstructure:"token_manager"` // ShareFolder is the location where to create shares in the recipient's storage provider. - ShareFolder string `mapstructure:"share_folder"` - DataTransfersFolder string `mapstructure:"data_transfers_folder"` - HomeMapping string `mapstructure:"home_mapping"` - TokenManagers map[string]map[string]interface{} `mapstructure:"token_managers"` - EtagCacheTTL int `mapstructure:"etag_cache_ttl"` - AllowedUserAgents map[string][]string `mapstructure:"allowed_user_agents"` // map[path][]user-agent - CreateHomeCacheTTL int `mapstructure:"create_home_cache_ttl"` + ShareFolder string `mapstructure:"share_folder"` + DataTransfersFolder string `mapstructure:"data_transfers_folder"` + HomeMapping string `mapstructure:"home_mapping"` + TokenManagers map[string]map[string]interface{} `mapstructure:"token_managers"` + AllowedUserAgents map[string][]string `mapstructure:"allowed_user_agents"` // map[path][]user-agent + CacheWarmupDriver string `mapstructure:"cache_warmup_driver"` + CacheWarmupDrivers map[string]map[string]interface{} `mapstructure:"cache_warmup_drivers"` + EtagCacheTTL int `mapstructure:"etag_cache_ttl"` + CreateHomeCacheTTL int `mapstructure:"create_home_cache_ttl"` + ResourceInfoCacheDriver string `mapstructure:"resource_info_cache_type"` + ResourceInfoCacheTTL int `mapstructure:"resource_info_cache_ttl"` + ResourceInfoCacheDrivers map[string]map[string]interface{} `mapstructure:"resource_info_caches"` } // sets defaults. @@ -116,11 +123,13 @@ func (c *config) ApplyDefaults() { } type svc struct { - c *config - dataGatewayURL url.URL - tokenmgr token.Manager - etagCache *ttlcache.Cache `mapstructure:"etag_cache"` - createHomeCache *ttlcache.Cache `mapstructure:"create_home_cache"` + c *config + dataGatewayURL url.URL + tokenmgr token.Manager + etagCache *ttlcache.Cache `mapstructure:"etag_cache"` + createHomeCache *ttlcache.Cache `mapstructure:"create_home_cache"` + resourceInfoCache cache.ResourceInfoCache + resourceInfoCacheTTL time.Duration } // New creates a new gateway svc that acts as a proxy for any grpc operation. @@ -151,12 +160,21 @@ func New(ctx context.Context, m map[string]interface{}) (rgrpc.Service, error) { _ = createHomeCache.SetTTL(time.Duration(c.CreateHomeCacheTTL) * time.Second) createHomeCache.SkipTTLExtensionOnHit(true) + rCache, _ := getCacheManager(c) + if c.ResourceInfoCacheTTL > 0 { + cwm, err := getCacheWarmupManager(c) + if err == nil { + go startCacheWarmup(cwm, rCache, c.ResourceInfoCacheTTL) + } + } + s := &svc{ - c: &c, - dataGatewayURL: *u, - tokenmgr: tokenManager, - etagCache: etagCache, - createHomeCache: createHomeCache, + c: &c, + dataGatewayURL: *u, + tokenmgr: tokenManager, + etagCache: etagCache, + createHomeCache: createHomeCache, + resourceInfoCache: rCache, } return s, nil @@ -217,3 +235,29 @@ func getTokenManager(manager string, m map[string]map[string]interface{}) (token return nil, errtypes.NotFound(fmt.Sprintf("driver %s not found for token manager", manager)) } + +func getCacheManager(c *config.Config) (cache.ResourceInfoCache, error) { + if f, ok := cachereg.NewFuncs[c.ResourceInfoCacheDriver]; ok { + return f(c.ResourceInfoCacheDrivers[c.ResourceInfoCacheDriver]) + } + return nil, fmt.Errorf("driver not found: %s", c.ResourceInfoCacheDriver) +} + +func getCacheWarmupManager(c *config.Config) (cache.Warmup, error) { + if f, ok := warmupreg.NewFuncs[c.CacheWarmupDriver]; ok { + return f(c.CacheWarmupDrivers[c.CacheWarmupDriver]) + } + return nil, fmt.Errorf("driver not found: %s", c.CacheWarmupDriver) +} + +func startCacheWarmup(cw cache.Warmup, rCache cache.ResourceInfoCache, ttl Duration) { + time.Sleep(2 * time.Second) + infos, err := cw.GetResourceInfos() + if err != nil { + return + } + for _, r := range infos { + key := resourceid.OwnCloudResourceIDWrap(r.Id) + _ = h.resourceInfoCache.SetWithExpire(key, r, h.resourceInfoCacheTTL) + } +} diff --git a/internal/grpc/services/gateway/usershareprovider.go b/internal/grpc/services/gateway/usershareprovider.go index fd60610394..0e0c3be531 100644 --- a/internal/grpc/services/gateway/usershareprovider.go +++ b/internal/grpc/services/gateway/usershareprovider.go @@ -328,6 +328,7 @@ func (s *svc) ListExistingReceivedShares(ctx context.Context, req *collaboration } // TODO(lopresti) incorporate the cache layer from internal/http/services/owncloud/ocs/handlers/apps/sharing/shares/shares.go + //s.resourceInfoCache stat, err := s.Stat(ctx, &provider.StatRequest{ Ref: &provider.Reference{ ResourceId: rs.Share.ResourceId, diff --git a/internal/http/services/owncloud/ocs/handlers/apps/sharing/shares/shares.go b/internal/http/services/owncloud/ocs/handlers/apps/sharing/shares/shares.go index 0af37ccfb6..97a2eeb467 100644 --- a/internal/http/services/owncloud/ocs/handlers/apps/sharing/shares/shares.go +++ b/internal/http/services/owncloud/ocs/handlers/apps/sharing/shares/shares.go @@ -48,7 +48,6 @@ import ( "github.com/cs3org/reva/internal/http/services/owncloud/ocs/conversions" "github.com/cs3org/reva/internal/http/services/owncloud/ocs/response" "github.com/cs3org/reva/pkg/appctx" - "github.com/cs3org/reva/pkg/spaces" "github.com/cs3org/reva/pkg/notification" "github.com/cs3org/reva/pkg/notification/notificationhelper" @@ -150,24 +149,18 @@ func (h *Handler) startCacheWarmup(c cache.Warmup) { } } -func (h *Handler) extractReference(r *http.Request) (*provider.Reference, error) { +func (h *Handler) extractReference(r *http.Request) (provider.Reference, error) { var ref provider.Reference - if spaceID := r.FormValue("space_ref"); spaceID != "" { - _, base, _, ok := spaces.DecodeResourceID(spaceID) - if !ok { - return nil, errors.New("bad space id format") - } - - ref.Path = base - } if p := r.FormValue("path"); p != "" { - if ref.Path == "" { - ref.Path = path.Join(h.homeNamespace, p) - } else { - ref.Path = path.Join(ref.Path, p) + ref = provider.Reference{Path: path.Join(h.homeNamespace, p)} + } else if spaceRef := r.FormValue("space_ref"); spaceRef != "" { + var err error + ref, err = utils.ParseStorageSpaceReference(spaceRef) + if err != nil { + return provider.Reference{}, err } } - return &ref, nil + return ref, nil } // CreateShare handles POST requests on /apps/files_sharing/api/v1/shares. @@ -193,7 +186,7 @@ func (h *Handler) CreateShare(w http.ResponseWriter, r *http.Request) { } statReq := provider.StatRequest{ - Ref: ref, + Ref: &ref, } log := appctx.GetLogger(ctx).With().Interface("ref", ref).Logger() @@ -1124,12 +1117,8 @@ func (h *Handler) addFilters(w http.ResponseWriter, r *http.Request, prefix stri return nil, nil, err } - target, err := h.extractReference(r) - if err != nil { - response.WriteOCSError(w, r, response.MetaServerError.StatusCode, "error extracting reference from request", err) - return nil, nil, err - } - info, status, err := h.getResourceInfoByPath(ctx, client, target.Path) + target := path.Join(prefix, r.FormValue("path")) + info, status, err := h.getResourceInfoByPath(ctx, client, target) if err != nil { response.WriteOCSError(w, r, response.MetaServerError.StatusCode, "error sending a grpc stat request", err) return nil, nil, err @@ -1152,10 +1141,6 @@ func (h *Handler) addFilters(w http.ResponseWriter, r *http.Request, prefix stri return collaborationFilters, linkFilters, nil } -func relativePathToSpaceID(info *provider.ResourceInfo) string { - return strings.TrimPrefix(info.Path, info.Id.SpaceId) -} - func (h *Handler) addFileInfo(ctx context.Context, s *conversions.ShareData, info *provider.ResourceInfo) error { log := appctx.GetLogger(ctx) if info != nil { @@ -1168,14 +1153,12 @@ func (h *Handler) addFileInfo(ctx context.Context, s *conversions.ShareData, inf s.MimeType = parsedMt // TODO STime: &types.Timestamp{Seconds: info.Mtime.Seconds, Nanos: info.Mtime.Nanos}, // TODO Storage: int - itemID := spaces.EncodeResourceID(info.Id) - - s.ItemSource = itemID + s.ItemSource = resourceid.OwnCloudResourceIDWrap(info.Id) s.FileSource = s.ItemSource switch { case h.sharePrefix == "/": - s.FileTarget = relativePathToSpaceID(info) - s.Path = relativePathToSpaceID(info) + s.FileTarget = info.Path + s.Path = info.Path case s.ShareType == conversions.ShareTypePublicLink: s.FileTarget = path.Join("/", path.Base(info.Path)) s.Path = path.Join("/", path.Base(info.Path)) @@ -1411,8 +1394,7 @@ func mapState(state collaboration.ShareState) int { var mapped int switch state { case collaboration.ShareState_SHARE_STATE_PENDING: - mapped = ocsStateAccepted - // mapped = ocsStatePending + mapped = ocsStatePending case collaboration.ShareState_SHARE_STATE_ACCEPTED: mapped = ocsStateAccepted case collaboration.ShareState_SHARE_STATE_REJECTED: