[#137] Add index page support #141

Merged
alexvanin merged 1 commit from nzinkevich/frostfs-http-gw:feature/137/index_page into master 2024-10-04 14:52:16 +00:00
Member

Added index HTML-page response for Get object endpoint if oid parameter is:

  1. empty
  2. is in S3 object name format (not ID) and is recognized as a path prefix

image

Added index HTML-page response for `Get object` endpoint if `oid` parameter is: 1) empty 2) is in `S3 object name` format (not ID) and is recognized as a path prefix ![image](/attachments/35766ca2-b4e3-4135-8bd6-3f83cdf085e6)
nzinkevich added 1 commit 2024-09-10 11:51:03 +00:00
[#137] Add index page support
All checks were successful
/ DCO (pull_request) Successful in 53s
/ Vulncheck (pull_request) Successful in 1m11s
/ Builds (pull_request) Successful in 1m6s
/ Lint (pull_request) Successful in 2m25s
/ Tests (pull_request) Successful in 1m4s
6b35eb4527
Signed-off-by: Nikita Zinkevich <n.zinkevich@yadro.com>
requested review from storage-services-developers 2024-09-10 11:51:35 +00:00
Owner

Could we consider implementing an option to specify the path for the index page template within the configuration settings?

Looking ahead, it would be beneficial to have the capability to retrieve templates directly from the container in future versions.

Could we consider implementing an option to specify the path for the index page template within the configuration settings? Looking ahead, it would be beneficial to have the capability to retrieve templates directly from the container in future versions.
nzinkevich force-pushed feature/137/index_page from 6b35eb4527 to 31b838f86a 2024-09-11 13:44:04 +00:00 Compare
requested review from storage-services-committers 2024-09-17 13:35:47 +00:00
alexvanin reviewed 2024-09-17 13:49:34 +00:00
@ -101,6 +101,11 @@ request_timeout: 5s # Timeout to check node health during rebalance.
rebalance_timer: 30s # Interval to check nodes health.
pool_error_threshold: 100 # The number of errors on connection after which node is considered as unhealthy.
# enable index page to see objects list for specified container and prefix
Owner
- # enable index page to see objects list for specified container and prefix
+ # Enable index page to see objects list for specified container and prefix
```diff - # enable index page to see objects list for specified container and prefix + # Enable index page to see objects list for specified container and prefix ```
alexvanin marked this conversation as resolved
@ -0,0 +68,4 @@
return
}
respObjects := make([]ResponseObject, len(nodes))
if err = json.Unmarshal(jsonNodes, &respObjects); err != nil {
Owner

Can we write direct converter from nodes to respObjects without JSON as a middle step?

Can we write direct converter from `nodes` to `respObjects` without JSON as a middle step?
alexvanin marked this conversation as resolved
@ -155,6 +157,8 @@ type Handler struct {
cache *cache.BucketCache
}
type ListType int
Owner

Unused?

Unused?
alexvanin marked this conversation as resolved
@ -50,3 +50,3 @@
AddedPathGetCidOid = "added path /get/{cid}/{oid}" // Info in ../../app.go
AddedPathGetByAttributeCidAttrKeyAttrVal = "added path /get_by_attribute/{cid}/{attr_key}/{attr_val:*}" // Info in ../../app.go
AddedPathZipCidPrefix = "added path /zip/{cid}/{prefix}" // Info in ../../app.go
AddedPathZipCidPrefix = "added path /zip/{cid}/{prefix:*}" // Info in ../../app.go
Owner

Can you add a small new issue on that? I am not sure this is the thing we need to fix, especially in this PR.

Codebase uses {prefix:*} but I am not sure this is required. Maybe we need to fix code.

Can you add a small new issue on that? I am not sure this is the thing we need to fix, especially in this PR. Codebase uses `{prefix:*}` but I am not sure this is required. Maybe we need to fix code.
alexvanin marked this conversation as resolved
tree/tree.go Outdated
@ -77,0 +82,4 @@
GetNodeID() []uint64
}
type SubTreeStream interface {
Owner

Unused?

Unused?
alexvanin marked this conversation as resolved
nzinkevich force-pushed feature/137/index_page from 31b838f86a to 82bd6f0f8e 2024-09-19 07:34:59 +00:00 Compare
nzinkevich force-pushed feature/137/index_page from 82bd6f0f8e to 5e24713418 2024-09-19 07:36:02 +00:00 Compare
alexvanin reviewed 2024-09-19 11:13:49 +00:00
tree/tree.go Outdated
@ -190,0 +292,4 @@
}
if len(intermediateNodes) == 0 {
return nil, layer.ErrNodeNotFound
Owner

I wonder, is it affected by an issue from TrueCloudLab/frostfs-s3-gw#488?

Should we adopt these s3 gateway changes here or it doesn't matter here?

I wonder, is it affected by an issue from https://git.frostfs.info/TrueCloudLab/frostfs-s3-gw/pulls/488? Should we adopt these s3 gateway changes here or it doesn't matter here?
Author
Member

I think it's okay because we don't extract err code from layer.error. Instead of that we check error and write to response a new one

if err != nil {

I think it's okay because we don't extract err code from layer.error. Instead of that we check error and write to response a new one https://git.frostfs.info/TrueCloudLab/frostfs-http-gw/src/commit/5e24713418c296261689e8da497e173cf8385a45/internal/handler/handler.go#L236
alexvanin marked this conversation as resolved
alexvanin reviewed 2024-09-19 11:17:26 +00:00
tree/tree.go Outdated
@ -160,8 +172,8 @@ func getLatestNode(nodes []NodeResponse) (NodeResponse, error) {
for i, node := range nodes {
currentCreationTime := node.GetTimestamp()
Owner

Can we synchronize this implementation with s3-gw?

We change this code anyway.

Can we synchronize this implementation with [s3-gw](https://git.frostfs.info/TrueCloudLab/frostfs-s3-gw/src/commit/34c1426b9f1d67fb7d39f597ca9ddcff840da888/pkg/service/tree/tree.go#L837)? We change this code anyway.
Author
Member

I think yes, they are quite similar

I think yes, they are quite similar
alexvanin marked this conversation as resolved
alexvanin reviewed 2024-09-19 11:22:54 +00:00
@ -20,3 +21,4 @@
return []uint64{n.response.GetNodeId()}
}
func (n GetNodeByPathResponseInfoWrapper) GetParentID() uint64 {
Owner

I suggest to synchronize this also with s3-gw and return []uint64. We did this to fix some tree service split issues. We would like to keep this code similar.

I suggest to synchronize this also with s3-gw and return `[]uint64`. We did this to fix some tree service split issues. We would like to keep this code similar.
alexvanin marked this conversation as resolved
alexvanin approved these changes 2024-09-19 11:26:08 +00:00
Dismissed
alexvanin left a comment
Owner

Overall this looks very pretty.

I suggest you to process comments or comment what you think about them.

Overall this looks very pretty. I suggest you to process comments or comment what you think about them.
nzinkevich force-pushed feature/137/index_page from 5e24713418 to 242bf356d3 2024-09-20 08:47:59 +00:00 Compare
nzinkevich dismissed alexvanin's review 2024-09-20 08:48:00 +00:00
Reason:

New commits pushed, approval review dismissed automatically according to repository settings

nzinkevich force-pushed feature/137/index_page from 242bf356d3 to 733362dbb2 2024-09-20 11:50:53 +00:00 Compare
nzinkevich force-pushed feature/137/index_page from 733362dbb2 to 7b07002269 2024-09-20 12:11:36 +00:00 Compare
nzinkevich added 1 commit 2024-09-20 12:22:50 +00:00
[#137] Add index page support
All checks were successful
/ DCO (pull_request) Successful in 1m8s
/ Builds (pull_request) Successful in 1m10s
/ Vulncheck (pull_request) Successful in 1m18s
/ Lint (pull_request) Successful in 2m4s
/ Tests (pull_request) Successful in 1m12s
b3f3c4cd99
Signed-off-by: Nikita Zinkevich <n.zinkevich@yadro.com>
nzinkevich force-pushed feature/137/index_page from b3f3c4cd99 to 21d9946f76 2024-09-20 13:27:22 +00:00 Compare
nzinkevich force-pushed feature/137/index_page from 21d9946f76 to 0c2f170252 2024-09-24 09:00:39 +00:00 Compare
alexvanin approved these changes 2024-09-24 12:38:04 +00:00
Dismissed
dkirillov reviewed 2024-09-24 14:11:02 +00:00
@ -192,2 +195,4 @@
v.SetDefault(cfgPoolErrorThreshold, defaultPoolErrorThreshold)
v.SetDefault(cfgIndexPageEnabled, false)
v.SetDefault(cfgIndexPageTemplatePath, "")
Member

Shouldn't we use template/index.gotmpl?

Shouldn't we use `template/index.gotmpl`?
dkirillov marked this conversation as resolved
@ -382,0 +390,4 @@
func (h *Handler) listObjects(ctx context.Context, bucketName, prefix string) ([]map[string]string, error) {
var (
log = h.log.With(zap.String("bucket", bucketName))
Member

It's better to write:

log := h.log.With(zap.String("bucket", bucketName))
It's better to write: ```golang log := h.log.With(zap.String("bucket", bucketName)) ```
dkirillov marked this conversation as resolved
nzinkevich force-pushed feature/137/index_page from 0c2f170252 to a8b3833813 2024-09-26 07:46:28 +00:00 Compare
nzinkevich dismissed alexvanin's review 2024-09-26 07:46:28 +00:00
Reason:

New commits pushed, approval review dismissed automatically according to repository settings

nzinkevich force-pushed feature/137/index_page from a8b3833813 to 26b36e442c 2024-09-26 08:18:58 +00:00 Compare
dkirillov reviewed 2024-09-26 12:02:41 +00:00
@ -0,0 +90,4 @@
func (h *Handler) browseObjects(c *fasthttp.RequestCtx, bucketName, prefix string) {
var log = h.log.With(zap.String("bucket", bucketName))
ctx := utils.GetContextFromRequest(c)
nodes, err := h.listObjects(ctx, bucketName, prefix)
Member

If len(nodes) == 0 probably we should return not found error

If `len(nodes) == 0` probably we should return not found error
Author
Member

I think we should return NotFound by default not only in that case but each time we browsing objects. Otherwise we would violate API

I think we should return NotFound by default not only in that case but each time we browsing objects. Otherwise we would violate API
Member

Could you clarify what do you mean?

By the way, currently we don't return not found if provided path ends with / and doesn't exist

Could you clarify what do you mean? By the way, currently we don't return `not found` if provided path ends with `/` and doesn't exist
Author
Member

I mean that index page returns html with status code 200 in places, where API should return 404. If so, enabling index page would cause API requests to return incorrect status codes.
At first I thought that all index pages should return 404.
But it turns out, that some folders would return 200. For example, if we create object with key /foo/bar/, request /get/bucketName/foo/bar/ returns 200, but /get/bucketName/foo/ returns 404 (because foo/ stored only in a tree service I suppose)

I mean that index page returns html with status code 200 in places, where API should return 404. If so, enabling index page would cause API requests to return incorrect status codes. At first I thought that all index pages should return 404. But it turns out, that some folders would return 200. For example, if we create object with key `/foo/bar/`, request `/get/bucketName/foo/bar/` returns 200, but `/get/bucketName/foo/` returns 404 (because `foo/` stored only in a tree service I suppose)
dkirillov reviewed 2024-09-26 12:07:36 +00:00
@ -96,0 +92,4 @@
zipCompression bool
clientCut bool
returnIndexPage bool
indexPageTemplatePath string
Member

Why do we need this field in appSettings?

Why do we need this field in `appSettings`?
dkirillov marked this conversation as resolved
dkirillov reviewed 2024-09-26 12:08:25 +00:00
@ -96,0 +93,4 @@
clientCut bool
returnIndexPage bool
indexPageTemplatePath string
indexPageTemplate string
Member

Why do we actually need this template be reloadable?

Why do we actually need this template be reloadable?
Author
Member
https://git.frostfs.info/TrueCloudLab/frostfs-http-gw/pulls/141#issuecomment-51270
Member

That comment doesn't answer about realoadability

That comment doesn't answer about realoadability
Author
Member

Why we should disable reloadability? Do you think that it's critical for reload performance?

Why we should disable reloadability? Do you think that it's critical for reload performance?
dkirillov reviewed 2024-09-26 12:10:43 +00:00
@ -104,0 +104,4 @@
# Enable index page to see objects list for specified container and prefix
index_page:
enabled: false
template_path: /templates/index.gotmpl
Member

Maybe it's better to provide path to real default config? internal/handler/templates/index.gotmpl

Maybe it's better to provide path to real default config? `internal/handler/templates/index.gotmpl`
dkirillov marked this conversation as resolved
dkirillov reviewed 2024-09-26 12:12:08 +00:00
@ -0,0 +18,4 @@
const (
dateFormat = "02-01-2006 15:04"
attrOID, attrCreated, attrFileName, attrSize = "OID", "Created", "FileName", "Size"
Member

Let's keep each constant on separate line

Let's keep each constant on separate line
dkirillov marked this conversation as resolved
dkirillov reviewed 2024-09-26 12:12:34 +00:00
@ -0,0 +88,4 @@
}
func (h *Handler) browseObjects(c *fasthttp.RequestCtx, bucketName, prefix string) {
var log = h.log.With(zap.String("bucket", bucketName))
Member

It's better to write:

log := h.log.With(zap.String("bucket", bucketName))
It's better to write: ```golang log := h.log.With(zap.String("bucket", bucketName)) ```
dkirillov marked this conversation as resolved
dkirillov reviewed 2024-09-26 12:13:10 +00:00
@ -39,0 +39,4 @@
func (n nodeResponse) GetNodeID() []uint64 {
return nil
}
func (n nodeResponse) GetParentID() []uint64 {
Member

Please add empty line between methods

Please add empty line between methods
Member

Still there isn't empty line

Still there isn't empty line
nzinkevich force-pushed feature/137/index_page from 26b36e442c to b25703aefa 2024-09-26 14:34:24 +00:00 Compare
nzinkevich was assigned by dkirillov 2024-09-27 06:35:30 +00:00
nzinkevich force-pushed feature/137/index_page from b25703aefa to 329f43c079 2024-09-27 06:47:36 +00:00 Compare
nzinkevich force-pushed feature/137/index_page from 329f43c079 to 585971ebf7 2024-09-30 05:19:28 +00:00 Compare
alexvanin requested changes 2024-09-30 08:36:24 +00:00
Dismissed
@ -0,0 +37,4 @@
)
//go:embed templates/index.gotmpl
var defaultTemplate string
Owner

I suggest to move default setting out of handler and put it into configuration of app.go

This way we will have single place in application that manages actual template string. It will be easier to manage it later.

What you think?

I suggest to move default setting out of handler and put it into configuration of app.go This way we will have single place in application that manages actual template string. It will be easier to manage it later. What you think?
Author
Member

If we move default template init to app.go, it wouldn't be so nice because go:embed allows only link files which is relative to current package (and prohibits using .. in path). Thus we should move (or at least symlink) template file to cmd/http-gw as well which I think we shouldn't do

If we move default template init to app.go, it wouldn't be so nice because `go:embed` allows only link files which is relative to current package (and prohibits using `..` in path). Thus we should move (or at least symlink) template file to cmd/http-gw as well which I think we shouldn't do
Member

But we still can introduce separate package for reading default template. And use it in app.go

But we still can introduce separate package for reading default template. And use it in app.go
alexvanin marked this conversation as resolved
@ -31,2 +31,4 @@
ZipCompression() bool
ClientCut() bool
IndexPageEnabled() bool
IndexPageTemplatePath() string
Owner

This interface method is unused. Remove it.

This interface method is unused. Remove it.
alexvanin marked this conversation as resolved
nzinkevich added 1 commit 2024-09-30 09:48:44 +00:00
[#137] Add index page support
Some checks failed
/ DCO (pull_request) Successful in 1m1s
/ Builds (pull_request) Successful in 55s
/ Lint (pull_request) Failing after 1m22s
/ Tests (pull_request) Successful in 57s
/ Vulncheck (pull_request) Successful in 1m28s
20f3ba6879
Signed-off-by: Nikita Zinkevich <n.zinkevich@yadro.com>
nzinkevich added 1 commit 2024-09-30 09:51:21 +00:00
[#137] Add index page support
Some checks failed
/ DCO (pull_request) Successful in 1m11s
/ Vulncheck (pull_request) Successful in 1m43s
/ Builds (pull_request) Successful in 56s
/ Lint (pull_request) Failing after 2m21s
/ Tests (pull_request) Successful in 1m12s
90d656d841
Signed-off-by: Nikita Zinkevich <n.zinkevich@yadro.com>
nzinkevich force-pushed feature/137/index_page from 90d656d841 to c57dec966b 2024-09-30 09:51:48 +00:00 Compare
nzinkevich force-pushed feature/137/index_page from c57dec966b to 48fb600ae2 2024-09-30 09:52:40 +00:00 Compare
nzinkevich force-pushed feature/137/index_page from 48fb600ae2 to 440df2ca7e 2024-09-30 11:21:47 +00:00 Compare
requested review from alexvanin 2024-10-01 05:28:18 +00:00
dkirillov reviewed 2024-10-01 09:29:50 +00:00
@ -0,0 +66,4 @@
{{end}}
{{else if and (not .FileName) .Size}}
{{/* this is an object with body of current dir */}}
..{{parentDir $prefix}}/
Member

It's quite confusing. At least it must be ../{{parentDir $prefix}}/. But it can be understood as object with key parent-dir/../parent-dir/ rather than parent-dir/

It's quite confusing. At least it must be `../{{parentDir $prefix}}/`. But it can be understood as object with key `parent-dir/../parent-dir/` rather than `parent-dir/`
Member

Please consider something like this

diff --git a/internal/handler/browse.go b/internal/handler/browse.go
index 67611a2..52b71a9 100644
--- a/internal/handler/browse.go
+++ b/internal/handler/browse.go
@@ -1,10 +1,10 @@
 package handler
 
 import (
-	"cmp"
 	_ "embed"
 	"html/template"
 	"net/url"
+	"sort"
 	"strconv"
 	"strings"
 	"time"
@@ -14,7 +14,6 @@ import (
 	"github.com/docker/go-units"
 	"github.com/valyala/fasthttp"
 	"go.uber.org/zap"
-	"golang.org/x/exp/slices"
 )
 
 const (
@@ -36,6 +35,7 @@ type (
 		Created  string
 		FileName string
 		Size     string
+		IsDir    bool
 	}
 )
 
@@ -57,6 +57,7 @@ func NewResponseObject(nodes map[string]string) ResponseObject {
 		Created:  nodes[attrCreated],
 		FileName: nodes[attrFileName],
 		Size:     nodes[attrSize],
+		IsDir:    nodes[attrOID] == "",
 	}
 }
 
@@ -70,6 +71,9 @@ func formatTimestamp(strdate string) string {
 }
 
 func formatSize(strsize string) string {
+	if strsize == "" {
+		return "0B"
+	}
 	size, err := strconv.ParseFloat(strsize, 64)
 	if err != nil {
 		return ""
@@ -127,24 +131,12 @@ func (h *Handler) browseObjects(c *fasthttp.RequestCtx, bucketInfo *data.BucketI
 		respObjects[i] = NewResponseObject(node)
 	}
 
-	slices.SortFunc(respObjects, func(a, b ResponseObject) int {
-		aIsDir := a.Size == ""
-		bIsDir := b.Size == ""
-
-		// return root object first
-		if a.FileName == "" {
-			return -1
-		} else if b.FileName == "" {
-			return 1
+	sort.Slice(respObjects, func(i, j int) bool {
+		if respObjects[i].IsDir == respObjects[j].IsDir {
+			return respObjects[i].FileName < respObjects[j].FileName
 		}
 
-		// dirs objects go first
-		if aIsDir && !bIsDir {
-			return -1
-		} else if !aIsDir && bIsDir {
-			return 1
-		}
-		return cmp.Compare(a.FileName, b.FileName)
+		return respObjects[i].IsDir
 	})
 
 	indexTemplate := h.config.IndexPageTemplate()
diff --git a/internal/handler/reader.go b/internal/handler/reader.go
index 65d258b..66067f1 100644
--- a/internal/handler/reader.go
+++ b/internal/handler/reader.go
@@ -53,6 +53,7 @@ func (h *Handler) receiveFile(ctx context.Context, req request, objectAddress oi
 		dis      = "inline"
 		start    = time.Now()
 		filename string
+		filepath string
 	)
 
 	prm := PrmObjectGet{
@@ -91,6 +92,8 @@ func (h *Handler) receiveFile(ctx context.Context, req request, objectAddress oi
 		switch key {
 		case object.AttributeFileName:
 			filename = val
+		case object.AttributeFilePath:
+			filepath = val
 		case object.AttributeTimestamp:
 			value, err := strconv.ParseInt(val, 10, 64)
 			if err != nil {
@@ -135,6 +138,10 @@ func (h *Handler) receiveFile(ctx context.Context, req request, objectAddress oi
 	}
 	req.SetContentType(contentType)
 
+	if filename == "" {
+		filename = filepath
+	}
+
 	req.Response.Header.Set(fasthttp.HeaderContentDisposition, dis+"; filename="+path.Base(filename))
 
 	req.Response.SetBodyStream(rObj.Payload, int(payloadSize))
diff --git a/internal/handler/templates/index.gotmpl b/internal/handler/templates/index.gotmpl
index 1626246..63ac7a3 100644
--- a/internal/handler/templates/index.gotmpl
+++ b/internal/handler/templates/index.gotmpl
@@ -34,6 +34,7 @@
         <th>Filename</th>
         <th>Size</th>
         <th>Created</th>
+        <th>Filepath</th>
         <th>Download</th>
     </tr>
     </thead>
@@ -41,42 +42,40 @@
     {{ $trimmedPrefix := trimPrefix $prefix }}
     {{if $trimmedPrefix }}
         <tr>
-            <td><a href="/get/{{$bucketName}}/{{ $trimmedPrefix }}/">../</a></td>
+            <td>⮐ <a href="/get/{{$bucketName}}/{{ $trimmedPrefix }}/">..</a></td>
+            <td></td>
             <td></td>
             <td></td>
         </tr>
     {{else}}
         <tr>
-            <td><a href="/get/{{ $bucketName}}/">../</a></td>
-            <td></td>
-            <td></td>
+            <td>⮐ <a href="/get/{{ $bucketName}}/">..</a></td>
+            <td>-</td>
+            <td>-</td>
+            <td>-</td>
+            <td>-</td>
         </tr>
     {{end}}
     {{range .Objects}}
         <tr>
             <td>
-                {{if and (not .FileName) (not .Size)}}
-                    {{if not .Created}}
-                        <a href="/get/{{ $bucketName}}/{{ urlencode $prefix .FileName .Size .Created }}">
-                            /
-                        </a>
-                    {{else}}
-                        {{/* current dir info */}}
-
-                    {{end}}
-                {{else if and (not .FileName) .Size}}
-                    {{/* this is an object with body of current dir */}}
-                    ..{{parentDir $prefix}}/
+                {{if .IsDir}}
+                    🗀
+                    <a href="/get/{{ $bucketName}}/{{ urlencode $prefix .FileName .Size .Created }}">
+                        {{.FileName}}/
+                    </a>
                 {{else}}
+                    🗎
                     <a href="/get/{{ $bucketName}}/{{ urlencode $prefix .FileName .Size .Created}}">
                         {{.FileName}}{{if and (not .Size) (not .Created)}}/{{end}}
                     </a>
                 {{end}}
             </td>
-            <td>{{ formatSize .Size }}</td>
-            <td>{{ formatTimestamp .Created }}</td>
+            <td> {{if not .IsDir}} {{ formatSize .Size }} {{end}}</td>
+            <td> {{if not .IsDir}} {{ formatTimestamp .Created }} {{end}}</td>
+            <td>{{ $prefix }}/{{ .FileName }}</td>
             <td>
-                {{ if .Size }}
+                {{ if not .IsDir }}
                     <a href="/get/{{ $bucketName}}/{{ urlencode $prefix .FileName .Size .Created }}?download=true">
                         Link
                     </a>

Please consider something like this ```diff diff --git a/internal/handler/browse.go b/internal/handler/browse.go index 67611a2..52b71a9 100644 --- a/internal/handler/browse.go +++ b/internal/handler/browse.go @@ -1,10 +1,10 @@ package handler import ( - "cmp" _ "embed" "html/template" "net/url" + "sort" "strconv" "strings" "time" @@ -14,7 +14,6 @@ import ( "github.com/docker/go-units" "github.com/valyala/fasthttp" "go.uber.org/zap" - "golang.org/x/exp/slices" ) const ( @@ -36,6 +35,7 @@ type ( Created string FileName string Size string + IsDir bool } ) @@ -57,6 +57,7 @@ func NewResponseObject(nodes map[string]string) ResponseObject { Created: nodes[attrCreated], FileName: nodes[attrFileName], Size: nodes[attrSize], + IsDir: nodes[attrOID] == "", } } @@ -70,6 +71,9 @@ func formatTimestamp(strdate string) string { } func formatSize(strsize string) string { + if strsize == "" { + return "0B" + } size, err := strconv.ParseFloat(strsize, 64) if err != nil { return "" @@ -127,24 +131,12 @@ func (h *Handler) browseObjects(c *fasthttp.RequestCtx, bucketInfo *data.BucketI respObjects[i] = NewResponseObject(node) } - slices.SortFunc(respObjects, func(a, b ResponseObject) int { - aIsDir := a.Size == "" - bIsDir := b.Size == "" - - // return root object first - if a.FileName == "" { - return -1 - } else if b.FileName == "" { - return 1 + sort.Slice(respObjects, func(i, j int) bool { + if respObjects[i].IsDir == respObjects[j].IsDir { + return respObjects[i].FileName < respObjects[j].FileName } - // dirs objects go first - if aIsDir && !bIsDir { - return -1 - } else if !aIsDir && bIsDir { - return 1 - } - return cmp.Compare(a.FileName, b.FileName) + return respObjects[i].IsDir }) indexTemplate := h.config.IndexPageTemplate() diff --git a/internal/handler/reader.go b/internal/handler/reader.go index 65d258b..66067f1 100644 --- a/internal/handler/reader.go +++ b/internal/handler/reader.go @@ -53,6 +53,7 @@ func (h *Handler) receiveFile(ctx context.Context, req request, objectAddress oi dis = "inline" start = time.Now() filename string + filepath string ) prm := PrmObjectGet{ @@ -91,6 +92,8 @@ func (h *Handler) receiveFile(ctx context.Context, req request, objectAddress oi switch key { case object.AttributeFileName: filename = val + case object.AttributeFilePath: + filepath = val case object.AttributeTimestamp: value, err := strconv.ParseInt(val, 10, 64) if err != nil { @@ -135,6 +138,10 @@ func (h *Handler) receiveFile(ctx context.Context, req request, objectAddress oi } req.SetContentType(contentType) + if filename == "" { + filename = filepath + } + req.Response.Header.Set(fasthttp.HeaderContentDisposition, dis+"; filename="+path.Base(filename)) req.Response.SetBodyStream(rObj.Payload, int(payloadSize)) diff --git a/internal/handler/templates/index.gotmpl b/internal/handler/templates/index.gotmpl index 1626246..63ac7a3 100644 --- a/internal/handler/templates/index.gotmpl +++ b/internal/handler/templates/index.gotmpl @@ -34,6 +34,7 @@ <th>Filename</th> <th>Size</th> <th>Created</th> + <th>Filepath</th> <th>Download</th> </tr> </thead> @@ -41,42 +42,40 @@ {{ $trimmedPrefix := trimPrefix $prefix }} {{if $trimmedPrefix }} <tr> - <td><a href="/get/{{$bucketName}}/{{ $trimmedPrefix }}/">../</a></td> + <td>⮐ <a href="/get/{{$bucketName}}/{{ $trimmedPrefix }}/">..</a></td> + <td></td> <td></td> <td></td> </tr> {{else}} <tr> - <td><a href="/get/{{ $bucketName}}/">../</a></td> - <td></td> - <td></td> + <td>⮐ <a href="/get/{{ $bucketName}}/">..</a></td> + <td>-</td> + <td>-</td> + <td>-</td> + <td>-</td> </tr> {{end}} {{range .Objects}} <tr> <td> - {{if and (not .FileName) (not .Size)}} - {{if not .Created}} - <a href="/get/{{ $bucketName}}/{{ urlencode $prefix .FileName .Size .Created }}"> - / - </a> - {{else}} - {{/* current dir info */}} - - {{end}} - {{else if and (not .FileName) .Size}} - {{/* this is an object with body of current dir */}} - ..{{parentDir $prefix}}/ + {{if .IsDir}} + 🗀 + <a href="/get/{{ $bucketName}}/{{ urlencode $prefix .FileName .Size .Created }}"> + {{.FileName}}/ + </a> {{else}} + 🗎 <a href="/get/{{ $bucketName}}/{{ urlencode $prefix .FileName .Size .Created}}"> {{.FileName}}{{if and (not .Size) (not .Created)}}/{{end}} </a> {{end}} </td> - <td>{{ formatSize .Size }}</td> - <td>{{ formatTimestamp .Created }}</td> + <td> {{if not .IsDir}} {{ formatSize .Size }} {{end}}</td> + <td> {{if not .IsDir}} {{ formatTimestamp .Created }} {{end}}</td> + <td>{{ $prefix }}/{{ .FileName }}</td> <td> - {{ if .Size }} + {{ if not .IsDir }} <a href="/get/{{ $bucketName}}/{{ urlencode $prefix .FileName .Size .Created }}?download=true"> Link </a> ```
Member

I'd like to see 0B if object has empty payload.

Also, please fix link with case like this image

objects:

  • /dir/
  • /dir/ //4
  • /dir/..
  • dir2/obj
  • tmp
  • tmp-raw
I'd like to see `0B` if object has empty payload. Also, please fix link with case like this ![image](/attachments/3e133776-08a0-4be4-8daa-5473db4fa5f1) objects: * `/dir/` * `/dir/ //4` * `/dir/..` * `dir2/obj` * `tmp` * `tmp-raw`
Member

Just wondering. Why did you drop perfect tiny back arrow icon (⮐)?

Just wondering. Why did you drop perfect tiny back arrow icon (⮐)?
nzinkevich force-pushed feature/137/index_page from 440df2ca7e to da9ce5f396 2024-10-02 05:30:19 +00:00 Compare
alexvanin approved these changes 2024-10-02 13:37:46 +00:00
Dismissed
dkirillov reviewed 2024-10-03 12:08:35 +00:00
@ -0,0 +33,4 @@
<tr>
<th>Filename</th>
<th>Size</th>
<th>Created</th>
Member

We must add fourth (Download for example)

We must add fourth <th> (`Download` for example)
nzinkevich force-pushed feature/137/index_page from da9ce5f396 to 8fe8f2dcc2 2024-10-04 11:26:08 +00:00 Compare
nzinkevich dismissed alexvanin's review 2024-10-04 11:26:09 +00:00
Reason:

New commits pushed, approval review dismissed automatically according to repository settings

alexvanin approved these changes 2024-10-04 14:52:11 +00:00
alexvanin merged commit 8fe8f2dcc2 into master 2024-10-04 14:52:16 +00:00
alexvanin deleted branch feature/137/index_page 2024-10-04 14:52:16 +00:00
Sign in to join this conversation.
No reviewers
TrueCloudLab/storage-services-developers
No milestone
No project
No assignees
4 participants
Notifications
Due date
The due date is invalid or out of range. Please use the format "yyyy-mm-dd".

No due date set.

Dependencies

No dependencies set.

Reference: TrueCloudLab/frostfs-http-gw#141
No description provided.