2015-03-24 06:24:00 +03:00
|
|
|
// Copyright 2015 Google Inc. 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 fuseops contains implementations of the fuse.Op interface that may
|
|
|
|
// be returned by fuse.Connection.ReadOp. See documentation in that package for
|
|
|
|
// more.
|
|
|
|
package fuseops
|
|
|
|
|
2015-03-24 06:36:44 +03:00
|
|
|
import (
|
2015-03-24 07:59:19 +03:00
|
|
|
"log"
|
|
|
|
"reflect"
|
2015-03-24 07:00:05 +03:00
|
|
|
"time"
|
|
|
|
|
2015-03-24 06:36:44 +03:00
|
|
|
"github.com/jacobsa/bazilfuse"
|
|
|
|
"golang.org/x/net/context"
|
|
|
|
)
|
2015-03-24 06:24:00 +03:00
|
|
|
|
2015-03-24 06:41:51 +03:00
|
|
|
// Convert the supplied bazilfuse request struct to an Op, returning nil if it
|
|
|
|
// is unknown.
|
2015-03-24 06:24:00 +03:00
|
|
|
//
|
|
|
|
// This function is an implementation detail of the fuse package, and must not
|
|
|
|
// be called by anyone else.
|
2015-03-24 07:59:19 +03:00
|
|
|
func Convert(r bazilfuse.Request, logger *log.Logger) (o Op) {
|
2015-03-24 06:36:44 +03:00
|
|
|
var co *commonOp
|
|
|
|
|
2015-03-24 06:48:42 +03:00
|
|
|
switch typed := r.(type) {
|
2015-03-24 06:36:44 +03:00
|
|
|
case *bazilfuse.InitRequest:
|
2015-03-24 06:48:42 +03:00
|
|
|
to := &InitOp{}
|
2015-03-24 06:36:44 +03:00
|
|
|
o = to
|
|
|
|
co = &to.commonOp
|
|
|
|
|
2015-03-24 06:48:19 +03:00
|
|
|
case *bazilfuse.LookupRequest:
|
2015-03-24 06:46:31 +03:00
|
|
|
to := &LookUpInodeOp{
|
2015-03-24 06:48:42 +03:00
|
|
|
Parent: InodeID(typed.Header.Node),
|
|
|
|
Name: typed.Name,
|
2015-03-24 06:46:31 +03:00
|
|
|
}
|
2015-03-24 06:40:58 +03:00
|
|
|
o = to
|
|
|
|
co = &to.commonOp
|
|
|
|
|
2015-03-24 06:48:19 +03:00
|
|
|
case *bazilfuse.GetattrRequest:
|
2015-03-24 06:46:31 +03:00
|
|
|
to := &GetInodeAttributesOp{
|
2015-03-24 06:49:27 +03:00
|
|
|
Inode: InodeID(typed.Header.Node),
|
2015-03-24 06:46:31 +03:00
|
|
|
}
|
2015-03-24 06:40:58 +03:00
|
|
|
o = to
|
|
|
|
co = &to.commonOp
|
|
|
|
|
2015-03-24 06:48:19 +03:00
|
|
|
case *bazilfuse.SetattrRequest:
|
2015-03-24 06:46:31 +03:00
|
|
|
to := &SetInodeAttributesOp{
|
2015-03-24 06:49:27 +03:00
|
|
|
Inode: InodeID(typed.Header.Node),
|
2015-03-24 06:46:31 +03:00
|
|
|
}
|
2015-03-24 06:40:58 +03:00
|
|
|
o = to
|
|
|
|
co = &to.commonOp
|
|
|
|
|
2015-03-24 06:48:19 +03:00
|
|
|
case *bazilfuse.MkdirRequest:
|
2015-03-24 06:46:31 +03:00
|
|
|
to := &MkDirOp{
|
2015-03-24 06:49:27 +03:00
|
|
|
Parent: InodeID(typed.Header.Node),
|
|
|
|
Name: typed.Name,
|
|
|
|
Mode: typed.Mode,
|
2015-03-24 06:46:31 +03:00
|
|
|
}
|
2015-03-24 06:40:58 +03:00
|
|
|
o = to
|
|
|
|
co = &to.commonOp
|
|
|
|
|
2015-03-24 06:48:19 +03:00
|
|
|
case *bazilfuse.CreateRequest:
|
2015-03-24 06:46:31 +03:00
|
|
|
to := &CreateFileOp{
|
2015-03-24 06:49:27 +03:00
|
|
|
Parent: InodeID(typed.Header.Node),
|
|
|
|
Name: typed.Name,
|
|
|
|
Mode: typed.Mode,
|
|
|
|
Flags: typed.Flags,
|
2015-03-24 06:46:31 +03:00
|
|
|
}
|
2015-03-24 06:40:58 +03:00
|
|
|
o = to
|
|
|
|
co = &to.commonOp
|
|
|
|
|
2015-03-24 06:48:19 +03:00
|
|
|
case *bazilfuse.RemoveRequest:
|
2015-03-24 06:51:07 +03:00
|
|
|
if typed.Dir {
|
|
|
|
to := &RmDirOp{
|
|
|
|
Parent: InodeID(typed.Header.Node),
|
|
|
|
Name: typed.Name,
|
|
|
|
}
|
|
|
|
o = to
|
|
|
|
co = &to.commonOp
|
|
|
|
} else {
|
|
|
|
to := &UnlinkOp{
|
|
|
|
Parent: InodeID(typed.Header.Node),
|
|
|
|
Name: typed.Name,
|
|
|
|
}
|
|
|
|
o = to
|
|
|
|
co = &to.commonOp
|
|
|
|
}
|
2015-03-24 06:40:58 +03:00
|
|
|
|
2015-03-24 06:48:19 +03:00
|
|
|
case *bazilfuse.OpenRequest:
|
2015-03-24 06:56:41 +03:00
|
|
|
if typed.Dir {
|
|
|
|
to := &OpenDirOp{
|
|
|
|
Inode: InodeID(typed.Header.Node),
|
|
|
|
Flags: typed.Flags,
|
|
|
|
}
|
|
|
|
o = to
|
|
|
|
co = &to.commonOp
|
|
|
|
} else {
|
|
|
|
to := &OpenFileOp{
|
|
|
|
Inode: InodeID(typed.Header.Node),
|
|
|
|
Flags: typed.Flags,
|
|
|
|
}
|
|
|
|
o = to
|
|
|
|
co = &to.commonOp
|
|
|
|
}
|
2015-03-24 06:40:58 +03:00
|
|
|
|
2015-03-24 06:48:19 +03:00
|
|
|
case *bazilfuse.ReadRequest:
|
2015-03-24 06:56:41 +03:00
|
|
|
if typed.Dir {
|
|
|
|
to := &ReadDirOp{
|
|
|
|
Inode: InodeID(typed.Header.Node),
|
|
|
|
Handle: HandleID(typed.Handle),
|
|
|
|
Offset: DirOffset(typed.Offset),
|
|
|
|
Size: typed.Size,
|
|
|
|
}
|
|
|
|
o = to
|
|
|
|
co = &to.commonOp
|
|
|
|
} else {
|
|
|
|
to := &ReadFileOp{
|
|
|
|
Inode: InodeID(typed.Header.Node),
|
|
|
|
Handle: HandleID(typed.Handle),
|
|
|
|
Offset: typed.Offset,
|
|
|
|
Size: typed.Size,
|
|
|
|
}
|
|
|
|
o = to
|
|
|
|
co = &to.commonOp
|
|
|
|
}
|
2015-03-24 06:40:58 +03:00
|
|
|
|
2015-03-24 06:48:19 +03:00
|
|
|
case *bazilfuse.ReleaseRequest:
|
2015-03-24 06:56:41 +03:00
|
|
|
if typed.Dir {
|
|
|
|
to := &ReleaseDirHandleOp{
|
|
|
|
Handle: HandleID(typed.Handle),
|
|
|
|
}
|
|
|
|
o = to
|
|
|
|
co = &to.commonOp
|
|
|
|
} else {
|
2015-03-24 08:02:40 +03:00
|
|
|
to := &ReleaseFileHandleOp{
|
2015-03-24 06:56:41 +03:00
|
|
|
Handle: HandleID(typed.Handle),
|
|
|
|
}
|
|
|
|
o = to
|
|
|
|
co = &to.commonOp
|
|
|
|
}
|
2015-03-24 06:40:58 +03:00
|
|
|
|
2015-03-24 06:48:19 +03:00
|
|
|
case *bazilfuse.WriteRequest:
|
2015-03-24 06:46:31 +03:00
|
|
|
to := &WriteFileOp{
|
2015-03-24 06:50:17 +03:00
|
|
|
Inode: InodeID(typed.Header.Node),
|
|
|
|
Handle: HandleID(typed.Handle),
|
|
|
|
Data: typed.Data,
|
|
|
|
Offset: typed.Offset,
|
2015-03-24 06:46:31 +03:00
|
|
|
}
|
2015-03-24 06:40:58 +03:00
|
|
|
o = to
|
|
|
|
co = &to.commonOp
|
|
|
|
|
2015-03-24 06:48:19 +03:00
|
|
|
case *bazilfuse.FsyncRequest:
|
2015-03-24 06:50:17 +03:00
|
|
|
// We don't currently support this for directories.
|
|
|
|
if typed.Dir {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-03-24 06:46:31 +03:00
|
|
|
to := &SyncFileOp{
|
2015-03-24 06:50:17 +03:00
|
|
|
Inode: InodeID(typed.Header.Node),
|
|
|
|
Handle: HandleID(typed.Handle),
|
2015-03-24 06:46:31 +03:00
|
|
|
}
|
2015-03-24 06:40:58 +03:00
|
|
|
o = to
|
|
|
|
co = &to.commonOp
|
|
|
|
|
2015-03-24 06:48:19 +03:00
|
|
|
case *bazilfuse.FlushRequest:
|
2015-03-24 06:46:31 +03:00
|
|
|
to := &FlushFileOp{
|
2015-03-24 06:50:17 +03:00
|
|
|
Inode: InodeID(typed.Header.Node),
|
|
|
|
Handle: HandleID(typed.Handle),
|
2015-03-24 06:46:31 +03:00
|
|
|
}
|
2015-03-24 06:40:58 +03:00
|
|
|
o = to
|
|
|
|
co = &to.commonOp
|
|
|
|
|
2015-03-24 06:36:44 +03:00
|
|
|
default:
|
2015-03-24 06:41:51 +03:00
|
|
|
return
|
2015-03-24 06:36:44 +03:00
|
|
|
}
|
|
|
|
|
2015-03-24 07:59:19 +03:00
|
|
|
co.init(reflect.TypeOf(o).String(), r, logger)
|
2015-03-24 06:36:44 +03:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-03-24 07:00:05 +03:00
|
|
|
func convertAttributes(inode InodeID, attr InodeAttributes) bazilfuse.Attr {
|
|
|
|
return bazilfuse.Attr{
|
|
|
|
Inode: uint64(inode),
|
|
|
|
Size: attr.Size,
|
|
|
|
Mode: attr.Mode,
|
|
|
|
Nlink: uint32(attr.Nlink),
|
|
|
|
Atime: attr.Atime,
|
|
|
|
Mtime: attr.Mtime,
|
|
|
|
Ctime: attr.Ctime,
|
|
|
|
Crtime: attr.Crtime,
|
|
|
|
Uid: attr.Uid,
|
|
|
|
Gid: attr.Gid,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Convert an absolute cache expiration time to a relative time from now for
|
|
|
|
// consumption by fuse.
|
|
|
|
func convertExpirationTime(t time.Time) (d time.Duration) {
|
|
|
|
// Fuse represents durations as unsigned 64-bit counts of seconds and 32-bit
|
|
|
|
// counts of nanoseconds (cf. http://goo.gl/EJupJV). The bazil.org/fuse
|
|
|
|
// package converts time.Duration values to this form in a straightforward
|
|
|
|
// way (cf. http://goo.gl/FJhV8j).
|
|
|
|
//
|
|
|
|
// So negative durations are right out. There is no need to cap the positive
|
|
|
|
// magnitude, because 2^64 seconds is well longer than the 2^63 ns range of
|
|
|
|
// time.Duration.
|
|
|
|
d = t.Sub(time.Now())
|
|
|
|
if d < 0 {
|
|
|
|
d = 0
|
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func convertChildInodeEntry(
|
|
|
|
in *ChildInodeEntry,
|
|
|
|
out *bazilfuse.LookupResponse) {
|
|
|
|
out.Node = bazilfuse.NodeID(in.Child)
|
|
|
|
out.Generation = uint64(in.Generation)
|
|
|
|
out.Attr = convertAttributes(in.Child, in.Attributes)
|
|
|
|
out.AttrValid = convertExpirationTime(in.AttributesExpiration)
|
|
|
|
out.EntryValid = convertExpirationTime(in.EntryExpiration)
|
|
|
|
}
|
|
|
|
|
2015-03-24 06:36:44 +03:00
|
|
|
// A helper for embedding common behavior.
|
|
|
|
type commonOp struct {
|
2015-03-24 07:59:19 +03:00
|
|
|
opType string
|
|
|
|
ctx context.Context
|
|
|
|
r bazilfuse.Request
|
|
|
|
logger *log.Logger
|
2015-03-24 06:36:44 +03:00
|
|
|
}
|
|
|
|
|
2015-03-24 07:59:19 +03:00
|
|
|
func (o *commonOp) init(
|
|
|
|
opType string,
|
|
|
|
r bazilfuse.Request,
|
|
|
|
logger *log.Logger) {
|
|
|
|
o.opType = opType
|
2015-03-24 06:38:18 +03:00
|
|
|
o.ctx = context.Background()
|
|
|
|
o.r = r
|
2015-03-24 07:59:19 +03:00
|
|
|
o.logger = logger
|
2015-03-24 06:38:18 +03:00
|
|
|
}
|
2015-03-24 06:36:44 +03:00
|
|
|
|
2015-03-24 06:38:18 +03:00
|
|
|
func (o *commonOp) Header() OpHeader {
|
|
|
|
bh := o.r.Hdr()
|
|
|
|
return OpHeader{
|
|
|
|
Uid: bh.Uid,
|
|
|
|
Gid: bh.Gid,
|
|
|
|
}
|
|
|
|
}
|
2015-03-24 06:36:44 +03:00
|
|
|
|
|
|
|
func (o *commonOp) Context() context.Context {
|
|
|
|
return o.ctx
|
|
|
|
}
|
|
|
|
|
2015-03-24 06:41:51 +03:00
|
|
|
func (o *commonOp) respondErr(err error) {
|
2015-03-24 07:48:18 +03:00
|
|
|
if err == nil {
|
2015-03-24 06:38:18 +03:00
|
|
|
panic("Expect non-nil here.")
|
|
|
|
}
|
|
|
|
|
2015-03-24 07:59:19 +03:00
|
|
|
o.logger.Printf(
|
|
|
|
"Responding with error to %s: %v",
|
|
|
|
o.opType,
|
|
|
|
err)
|
|
|
|
|
2015-03-24 06:38:18 +03:00
|
|
|
o.r.RespondError(err)
|
|
|
|
}
|