refactor: Rewrite path sanitizer to support write operations on relative tar archives
This commit is contained in:
@@ -349,6 +349,9 @@ func (f *File) Name() string {
|
|||||||
"name": f.name,
|
"name": f.name,
|
||||||
})
|
})
|
||||||
|
|
||||||
|
f.ioLock.Lock()
|
||||||
|
defer f.ioLock.Unlock()
|
||||||
|
|
||||||
return f.name
|
return f.name
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -357,6 +360,9 @@ func (f *File) Stat() (os.FileInfo, error) {
|
|||||||
"name": f.name,
|
"name": f.name,
|
||||||
})
|
})
|
||||||
|
|
||||||
|
f.ioLock.Lock()
|
||||||
|
defer f.ioLock.Unlock()
|
||||||
|
|
||||||
return f.info, nil
|
return f.info, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -366,6 +372,9 @@ func (f *File) Readdir(count int) ([]os.FileInfo, error) {
|
|||||||
"count": count,
|
"count": count,
|
||||||
})
|
})
|
||||||
|
|
||||||
|
f.ioLock.Lock()
|
||||||
|
defer f.ioLock.Unlock()
|
||||||
|
|
||||||
hdrs, err := inventory.List(
|
hdrs, err := inventory.List(
|
||||||
f.metadata,
|
f.metadata,
|
||||||
|
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ package fs
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"archive/tar"
|
"archive/tar"
|
||||||
|
"context"
|
||||||
"database/sql"
|
"database/sql"
|
||||||
"io"
|
"io"
|
||||||
"os"
|
"os"
|
||||||
@@ -70,6 +71,9 @@ func (f *STFS) Name() string {
|
|||||||
"name": config.FileSystemNameSTFS,
|
"name": config.FileSystemNameSTFS,
|
||||||
})
|
})
|
||||||
|
|
||||||
|
f.ioLock.Lock()
|
||||||
|
defer f.ioLock.Unlock()
|
||||||
|
|
||||||
return config.FileSystemNameSTFS
|
return config.FileSystemNameSTFS
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -170,7 +174,16 @@ func (f *STFS) MkdirRoot(name string, perm os.FileMode) error {
|
|||||||
f.ioLock.Lock()
|
f.ioLock.Lock()
|
||||||
defer f.ioLock.Unlock()
|
defer f.ioLock.Unlock()
|
||||||
|
|
||||||
return f.mknodeWithoutLocking(true, name, perm, true, "")
|
if err := f.mknodeWithoutLocking(true, name, perm, true, ""); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ensure that the new root path is being used
|
||||||
|
if _, err := f.metadata.Metadata.GetRootPath(context.Background()); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (f *STFS) Mkdir(name string, perm os.FileMode) error {
|
func (f *STFS) Mkdir(name string, perm os.FileMode) error {
|
||||||
|
|||||||
@@ -28,7 +28,10 @@ type depth struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type MetadataPersister struct {
|
type MetadataPersister struct {
|
||||||
*ipersisters.SQLite
|
sqlite *ipersisters.SQLite
|
||||||
|
|
||||||
|
root string
|
||||||
|
rootIsEmptyString bool
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewMetadataPersister(dbPath string) *MetadataPersister {
|
func NewMetadataPersister(dbPath string) *MetadataPersister {
|
||||||
@@ -41,30 +44,49 @@ func NewMetadataPersister(dbPath string) *MetadataPersister {
|
|||||||
Dir: "../../db/sqlite/migrations/metadata",
|
Dir: "../../db/sqlite/migrations/metadata",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
"",
|
||||||
|
false,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (p *MetadataPersister) Open() error {
|
||||||
|
if err := p.sqlite.Open(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
root, err := p.GetRootPath(context.Background())
|
||||||
|
|
||||||
|
// Ignore if root directory can't be found, which can happen i.e. on initial archiving
|
||||||
|
if err == config.ErrNoRootDirectory {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
p.root = root
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func (p *MetadataPersister) UpsertHeader(ctx context.Context, dbhdr *config.Header) error {
|
func (p *MetadataPersister) UpsertHeader(ctx context.Context, dbhdr *config.Header) error {
|
||||||
hdr := converters.ConfigHeaderToDBHeader(dbhdr)
|
idbhdr := converters.ConfigHeaderToDBHeader(dbhdr)
|
||||||
|
|
||||||
if _, err := models.FindHeader(ctx, p.DB, hdr.Name, models.HeaderColumns.Name); err != nil {
|
hdr := *idbhdr
|
||||||
|
hdr.Name = p.getSanitizedPath(ctx, idbhdr.Name)
|
||||||
|
|
||||||
|
if _, err := models.FindHeader(ctx, p.sqlite.DB, hdr.Name, models.HeaderColumns.Name); err != nil {
|
||||||
if err == sql.ErrNoRows {
|
if err == sql.ErrNoRows {
|
||||||
if _, err := models.FindHeader(ctx, p.DB, p.withRelativeRoot(ctx, hdr.Name), models.HeaderColumns.Name); err == nil {
|
if err := hdr.Insert(ctx, p.sqlite.DB, boil.Infer()); err != nil {
|
||||||
hdr.Name = p.withRelativeRoot(ctx, hdr.Name)
|
return err
|
||||||
} else {
|
|
||||||
if err := hdr.Insert(ctx, p.DB, boil.Infer()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if _, err := hdr.Update(ctx, p.DB, boil.Infer()); err != nil {
|
if _, err := hdr.Update(ctx, p.sqlite.DB, boil.Infer()); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -74,23 +96,20 @@ func (p *MetadataPersister) UpsertHeader(ctx context.Context, dbhdr *config.Head
|
|||||||
func (p *MetadataPersister) UpdateHeaderMetadata(ctx context.Context, dbhdr *config.Header) error {
|
func (p *MetadataPersister) UpdateHeaderMetadata(ctx context.Context, dbhdr *config.Header) error {
|
||||||
idbhdr := converters.ConfigHeaderToDBHeader(dbhdr)
|
idbhdr := converters.ConfigHeaderToDBHeader(dbhdr)
|
||||||
|
|
||||||
if _, err := idbhdr.Update(ctx, p.DB, boil.Infer()); err != nil {
|
hdr := *idbhdr
|
||||||
if err == sql.ErrNoRows {
|
hdr.Name = p.getSanitizedPath(ctx, idbhdr.Name)
|
||||||
hdr := *idbhdr
|
|
||||||
hdr.Name = p.withRelativeRoot(ctx, idbhdr.Name)
|
|
||||||
|
|
||||||
if _, err := hdr.Update(ctx, p.DB, boil.Infer()); err != nil {
|
if _, err := hdr.Update(ctx, p.sqlite.DB, boil.Infer()); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
|
||||||
} else {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *MetadataPersister) MoveHeader(ctx context.Context, oldName string, newName string, lastknownrecord, lastknownblock int64) error {
|
func (p *MetadataPersister) MoveHeader(ctx context.Context, oldName string, newName string, lastknownrecord, lastknownblock int64) error {
|
||||||
|
newName = p.getSanitizedPath(ctx, newName)
|
||||||
|
oldName = p.getSanitizedPath(ctx, oldName)
|
||||||
|
|
||||||
// We can't do this with `dbhdr.Update` because we are renaming the primary key
|
// We can't do this with `dbhdr.Update` because we are renaming the primary key
|
||||||
n, err := queries.Raw(
|
n, err := queries.Raw(
|
||||||
fmt.Sprintf(
|
fmt.Sprintf(
|
||||||
@@ -105,7 +124,7 @@ func (p *MetadataPersister) MoveHeader(ctx context.Context, oldName string, newN
|
|||||||
lastknownrecord,
|
lastknownrecord,
|
||||||
lastknownblock,
|
lastknownblock,
|
||||||
oldName,
|
oldName,
|
||||||
).ExecContext(ctx, p.DB)
|
).ExecContext(ctx, p.sqlite.DB)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@@ -120,16 +139,16 @@ func (p *MetadataPersister) MoveHeader(ctx context.Context, oldName string, newN
|
|||||||
fmt.Sprintf(
|
fmt.Sprintf(
|
||||||
`update %v set %v = ?, %v = ?, %v = ? where %v = ?;`,
|
`update %v set %v = ?, %v = ?, %v = ? where %v = ?;`,
|
||||||
models.TableNames.Headers,
|
models.TableNames.Headers,
|
||||||
p.withRelativeRoot(ctx, models.HeaderColumns.Name),
|
models.HeaderColumns.Name,
|
||||||
models.HeaderColumns.Lastknownrecord,
|
models.HeaderColumns.Lastknownrecord,
|
||||||
models.HeaderColumns.Lastknownblock,
|
models.HeaderColumns.Lastknownblock,
|
||||||
p.withRelativeRoot(ctx, models.HeaderColumns.Name),
|
models.HeaderColumns.Name,
|
||||||
),
|
),
|
||||||
newName,
|
newName,
|
||||||
lastknownrecord,
|
lastknownrecord,
|
||||||
lastknownblock,
|
lastknownblock,
|
||||||
oldName,
|
oldName,
|
||||||
).ExecContext(ctx, p.DB); err != nil {
|
).ExecContext(ctx, p.sqlite.DB); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -140,7 +159,7 @@ func (p *MetadataPersister) MoveHeader(ctx context.Context, oldName string, newN
|
|||||||
func (p *MetadataPersister) GetHeaders(ctx context.Context) ([]*config.Header, error) {
|
func (p *MetadataPersister) GetHeaders(ctx context.Context) ([]*config.Header, error) {
|
||||||
dbhdrs, err := models.Headers(
|
dbhdrs, err := models.Headers(
|
||||||
qm.Where(models.HeaderColumns.Deleted+" != 1"),
|
qm.Where(models.HeaderColumns.Deleted+" != 1"),
|
||||||
).All(ctx, p.DB)
|
).All(ctx, p.sqlite.DB)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return []*config.Header{}, err
|
return []*config.Header{}, err
|
||||||
}
|
}
|
||||||
@@ -154,46 +173,30 @@ func (p *MetadataPersister) GetHeaders(ctx context.Context) ([]*config.Header, e
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (p *MetadataPersister) GetHeader(ctx context.Context, name string) (*config.Header, error) {
|
func (p *MetadataPersister) GetHeader(ctx context.Context, name string) (*config.Header, error) {
|
||||||
|
name = p.getSanitizedPath(ctx, name)
|
||||||
|
|
||||||
hdr, err := models.Headers(
|
hdr, err := models.Headers(
|
||||||
qm.Where(models.HeaderColumns.Name+" = ?", name),
|
qm.Where(models.HeaderColumns.Name+" = ?", name),
|
||||||
qm.Where(models.HeaderColumns.Deleted+" != 1"),
|
qm.Where(models.HeaderColumns.Deleted+" != 1"),
|
||||||
).One(ctx, p.DB)
|
).One(ctx, p.sqlite.DB)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if err == sql.ErrNoRows {
|
return nil, err
|
||||||
hdr, err = models.Headers(
|
|
||||||
qm.Where(models.HeaderColumns.Name+" = ?", p.withRelativeRoot(ctx, name)),
|
|
||||||
qm.Where(models.HeaderColumns.Deleted+" != 1"),
|
|
||||||
).One(ctx, p.DB)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return converters.DBHeaderToConfigHeader(hdr), nil
|
return converters.DBHeaderToConfigHeader(hdr), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *MetadataPersister) GetHeaderChildren(ctx context.Context, name string) ([]*config.Header, error) {
|
func (p *MetadataPersister) GetHeaderChildren(ctx context.Context, name string) ([]*config.Header, error) {
|
||||||
|
name = p.getSanitizedPath(ctx, name)
|
||||||
|
|
||||||
headers, err := models.Headers(
|
headers, err := models.Headers(
|
||||||
qm.Where(models.HeaderColumns.Name+" like ?", strings.TrimSuffix(name, "/")+"/%"), // Prevent double trailing slashes
|
qm.Where(models.HeaderColumns.Name+" like ?", strings.TrimSuffix(name, "/")+"/%"), // Prevent double trailing slashes
|
||||||
qm.Where(models.HeaderColumns.Deleted+" != 1"),
|
qm.Where(models.HeaderColumns.Deleted+" != 1"),
|
||||||
).All(ctx, p.DB)
|
).All(ctx, p.sqlite.DB)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(headers) < 1 {
|
|
||||||
headers, err = models.Headers(
|
|
||||||
qm.Where(models.HeaderColumns.Name+" like ?", p.withRelativeRoot(ctx, strings.TrimSuffix(name, "/")+"/%")), // Prevent double trailing slashes
|
|
||||||
qm.Where(models.HeaderColumns.Deleted+" != 1"),
|
|
||||||
).All(ctx, p.DB)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
outhdrs := []*config.Header{}
|
outhdrs := []*config.Header{}
|
||||||
for _, hdr := range headers {
|
for _, hdr := range headers {
|
||||||
prefix := strings.TrimSuffix(hdr.Name, "/")
|
prefix := strings.TrimSuffix(hdr.Name, "/")
|
||||||
@@ -206,6 +209,11 @@ func (p *MetadataPersister) GetHeaderChildren(ctx context.Context, name string)
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (p *MetadataPersister) GetRootPath(ctx context.Context) (string, error) {
|
func (p *MetadataPersister) GetRootPath(ctx context.Context) (string, error) {
|
||||||
|
// Cache the root directory
|
||||||
|
if p.root != "" {
|
||||||
|
return p.root, nil
|
||||||
|
}
|
||||||
|
|
||||||
root := models.Header{}
|
root := models.Header{}
|
||||||
|
|
||||||
if err := queries.Raw(
|
if err := queries.Raw(
|
||||||
@@ -216,7 +224,7 @@ func (p *MetadataPersister) GetRootPath(ctx context.Context) (string, error) {
|
|||||||
models.TableNames.Headers,
|
models.TableNames.Headers,
|
||||||
models.HeaderColumns.Deleted,
|
models.HeaderColumns.Deleted,
|
||||||
),
|
),
|
||||||
).Bind(ctx, p.DB, &root); err != nil {
|
).Bind(ctx, p.sqlite.DB, &root); err != nil {
|
||||||
if strings.Contains(err.Error(), "converting NULL to string is unsupported") {
|
if strings.Contains(err.Error(), "converting NULL to string is unsupported") {
|
||||||
return "", config.ErrNoRootDirectory
|
return "", config.ErrNoRootDirectory
|
||||||
}
|
}
|
||||||
@@ -224,10 +232,13 @@ func (p *MetadataPersister) GetRootPath(ctx context.Context) (string, error) {
|
|||||||
return "", err
|
return "", err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
p.root = root.Name
|
||||||
|
|
||||||
return root.Name, nil
|
return root.Name, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *MetadataPersister) GetHeaderDirectChildren(ctx context.Context, name string, limit int) ([]*config.Header, error) {
|
func (p *MetadataPersister) GetHeaderDirectChildren(ctx context.Context, name string, limit int) ([]*config.Header, error) {
|
||||||
|
name = p.getSanitizedPath(ctx, name)
|
||||||
prefix := strings.TrimSuffix(name, "/") + "/"
|
prefix := strings.TrimSuffix(name, "/") + "/"
|
||||||
rootDepth := 0
|
rootDepth := 0
|
||||||
headers := []*config.Header{}
|
headers := []*config.Header{}
|
||||||
@@ -245,7 +256,7 @@ func (p *MetadataPersister) GetHeaderDirectChildren(ctx context.Context, name st
|
|||||||
models.TableNames.Headers,
|
models.TableNames.Headers,
|
||||||
models.HeaderColumns.Deleted,
|
models.HeaderColumns.Deleted,
|
||||||
),
|
),
|
||||||
).Bind(ctx, p.DB, &depth); err != nil {
|
).Bind(ctx, p.sqlite.DB, &depth); err != nil {
|
||||||
if err == sql.ErrNoRows {
|
if err == sql.ErrNoRows {
|
||||||
return headers, nil
|
return headers, nil
|
||||||
}
|
}
|
||||||
@@ -310,7 +321,7 @@ where %v like ?
|
|||||||
rootDepth,
|
rootDepth,
|
||||||
rootDepth+1,
|
rootDepth+1,
|
||||||
limit+1, // +1 to accomodate the parent directory if it exists
|
limit+1, // +1 to accomodate the parent directory if it exists
|
||||||
).Bind(ctx, p.DB, &headers); err != nil {
|
).Bind(ctx, p.sqlite.DB, &headers); err != nil {
|
||||||
if err == sql.ErrNoRows {
|
if err == sql.ErrNoRows {
|
||||||
return headers, nil
|
return headers, nil
|
||||||
}
|
}
|
||||||
@@ -326,7 +337,7 @@ where %v like ?
|
|||||||
prefix+"%",
|
prefix+"%",
|
||||||
rootDepth,
|
rootDepth,
|
||||||
rootDepth+1,
|
rootDepth+1,
|
||||||
).Bind(ctx, p.DB, &headers); err != nil {
|
).Bind(ctx, p.sqlite.DB, &headers); err != nil {
|
||||||
if err == sql.ErrNoRows {
|
if err == sql.ErrNoRows {
|
||||||
return headers, nil
|
return headers, nil
|
||||||
}
|
}
|
||||||
@@ -339,14 +350,7 @@ where %v like ?
|
|||||||
|
|
||||||
headers, err := getHeaders(prefix)
|
headers, err := getHeaders(prefix)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
headers, err = getHeaders(p.withRelativeRoot(ctx, prefix))
|
return nil, err
|
||||||
if err == sql.ErrNoRows {
|
|
||||||
return headers, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
outhdrs := []*config.Header{}
|
outhdrs := []*config.Header{}
|
||||||
@@ -365,27 +369,18 @@ where %v like ?
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (p *MetadataPersister) DeleteHeader(ctx context.Context, name string, lastknownrecord, lastknownblock int64) (*config.Header, error) {
|
func (p *MetadataPersister) DeleteHeader(ctx context.Context, name string, lastknownrecord, lastknownblock int64) (*config.Header, error) {
|
||||||
hdr, err := models.FindHeader(ctx, p.DB, name)
|
name = p.getSanitizedPath(ctx, name)
|
||||||
if err != nil {
|
|
||||||
if err == sql.ErrNoRows {
|
|
||||||
hdr, err = models.FindHeader(ctx, p.DB, p.withRelativeRoot(ctx, name))
|
|
||||||
if err == sql.ErrNoRows {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
if err != nil {
|
hdr, err := models.FindHeader(ctx, p.sqlite.DB, name)
|
||||||
return nil, err
|
if err != nil {
|
||||||
}
|
return nil, err
|
||||||
} else {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
hdr.Deleted = 1
|
hdr.Deleted = 1
|
||||||
hdr.Lastknownrecord = lastknownrecord
|
hdr.Lastknownrecord = lastknownrecord
|
||||||
hdr.Lastknownblock = lastknownblock
|
hdr.Lastknownblock = lastknownblock
|
||||||
|
|
||||||
if _, err := hdr.Update(ctx, p.DB, boil.Infer()); err != nil {
|
if _, err := hdr.Update(ctx, p.sqlite.DB, boil.Infer()); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -404,7 +399,7 @@ func (p *MetadataPersister) GetLastIndexedRecordAndBlock(ctx context.Context, re
|
|||||||
models.TableNames.Headers,
|
models.TableNames.Headers,
|
||||||
),
|
),
|
||||||
recordSize,
|
recordSize,
|
||||||
).Bind(ctx, p.DB, &header); err != nil {
|
).Bind(ctx, p.sqlite.DB, &header); err != nil {
|
||||||
if err == sql.ErrNoRows {
|
if err == sql.ErrNoRows {
|
||||||
return 0, 0, nil
|
return 0, 0, nil
|
||||||
}
|
}
|
||||||
@@ -416,10 +411,13 @@ func (p *MetadataPersister) GetLastIndexedRecordAndBlock(ctx context.Context, re
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (p *MetadataPersister) PurgeAllHeaders(ctx context.Context) error {
|
func (p *MetadataPersister) PurgeAllHeaders(ctx context.Context) error {
|
||||||
if _, err := models.Headers().DeleteAll(ctx, p.DB); err != nil {
|
if _, err := models.Headers().DeleteAll(ctx, p.sqlite.DB); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
p.root = ""
|
||||||
|
p.rootIsEmptyString = false
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -427,7 +425,7 @@ func (p *MetadataPersister) headerExistsExact(ctx context.Context, name string)
|
|||||||
exists, err := models.Headers(
|
exists, err := models.Headers(
|
||||||
qm.Where(models.HeaderColumns.Name+" = ?", name),
|
qm.Where(models.HeaderColumns.Name+" = ?", name),
|
||||||
qm.Where(models.HeaderColumns.Deleted+" != 1"),
|
qm.Where(models.HeaderColumns.Deleted+" != 1"),
|
||||||
).Exists(ctx, p.DB)
|
).Exists(ctx, p.sqlite.DB)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@@ -439,26 +437,39 @@ func (p *MetadataPersister) headerExistsExact(ctx context.Context, name string)
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *MetadataPersister) withRelativeRoot(ctx context.Context, root string) string {
|
func (p *MetadataPersister) getSanitizedPath(ctx context.Context, name string) string {
|
||||||
|
// If root is queried, return actual root
|
||||||
|
if pathext.IsRoot(name) {
|
||||||
|
return p.root
|
||||||
|
}
|
||||||
|
|
||||||
|
// If root has not been set, the incoming path is absolute and no header with the exact name "" (empty string) exists, assume it is root
|
||||||
|
if p.root == "" && strings.HasPrefix(name, "/") && !p.rootIsEmptyString {
|
||||||
|
if err := p.headerExistsExact(ctx, ""); err != nil {
|
||||||
|
p.root = name
|
||||||
|
|
||||||
|
return p.root
|
||||||
|
} else {
|
||||||
|
p.rootIsEmptyString = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Keep absolute paths untouched if root is also absolute
|
||||||
|
if strings.HasPrefix(p.root, "/") && strings.HasPrefix(name, "/") {
|
||||||
|
return name
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get correct root prefix
|
||||||
prefix := ""
|
prefix := ""
|
||||||
if err := p.headerExistsExact(ctx, ""); err == nil {
|
if p.root == "" {
|
||||||
prefix = ""
|
prefix = ""
|
||||||
} else if err := p.headerExistsExact(ctx, "."); err == nil {
|
} else if p.root == "." {
|
||||||
prefix = "."
|
prefix = "."
|
||||||
} else if err := p.headerExistsExact(ctx, "/"); err == nil {
|
} else if p.root == "/" {
|
||||||
prefix = "/"
|
prefix = "/"
|
||||||
} else {
|
} else {
|
||||||
prefix = "./" // Special case: There is no root directory, only files, and the files start with `./`
|
return "./" + filepath.Clean(strings.TrimPrefix(name, "/")) // Special case: There is no root directory, only files, and the files start with `./`
|
||||||
}
|
}
|
||||||
|
|
||||||
if pathext.IsRoot(root) {
|
return path.Join(prefix, strings.TrimPrefix(name, "/"))
|
||||||
return prefix
|
|
||||||
}
|
|
||||||
|
|
||||||
if prefix == "./" {
|
|
||||||
// Special case: There is no root directory, only files, and the files start with `./`; we can't do path.Join, as `./asdf.txt` would be shortened to `asdf.txt`
|
|
||||||
return prefix + filepath.Clean(strings.TrimPrefix(root, "/"))
|
|
||||||
}
|
|
||||||
|
|
||||||
return path.Join(prefix, filepath.Clean(strings.TrimPrefix(root, "/")))
|
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user