2015-05-08 09:28:53 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2015 Alex Yatskov <alex@foosoft.net>
|
|
|
|
* Author: Alex Yatskov <alex@foosoft.net>
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy of
|
|
|
|
* this software and associated documentation files (the "Software"), to deal in
|
|
|
|
* the Software without restriction, including without limitation the rights to
|
|
|
|
* use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
|
|
|
|
* the Software, and to permit persons to whom the Software is furnished to do so,
|
|
|
|
* subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in all
|
|
|
|
* copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
|
|
|
|
* FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
|
|
|
|
* COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
|
|
|
|
* IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
|
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
2015-05-08 09:01:24 +00:00
|
|
|
package main
|
|
|
|
|
2015-05-09 04:59:38 +00:00
|
|
|
import (
|
2015-05-12 10:48:11 +00:00
|
|
|
"bazil.org/fuse/fs"
|
2015-05-10 05:14:15 +00:00
|
|
|
"fmt"
|
2015-05-10 07:00:35 +00:00
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2015-05-10 05:14:15 +00:00
|
|
|
"regexp"
|
|
|
|
"strconv"
|
2015-05-14 06:10:42 +00:00
|
|
|
"strings"
|
2015-05-09 04:59:38 +00:00
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
2015-05-10 07:15:17 +00:00
|
|
|
type version struct {
|
2015-05-09 13:48:08 +00:00
|
|
|
base string
|
2015-05-10 07:15:17 +00:00
|
|
|
parent *version
|
2015-05-09 13:48:08 +00:00
|
|
|
timestamp time.Time
|
2015-05-15 04:31:57 +00:00
|
|
|
meta *versionMetadata
|
2015-05-14 05:24:52 +00:00
|
|
|
root *versionedDir
|
2015-05-16 07:50:15 +00:00
|
|
|
inodeAloc InodeAllocator
|
2015-05-09 04:59:38 +00:00
|
|
|
}
|
|
|
|
|
2015-05-16 07:50:15 +00:00
|
|
|
type InodeAllocator interface {
|
|
|
|
AllocInode() uint64
|
|
|
|
}
|
|
|
|
|
|
|
|
func newVersion(base string, allocator InodeAllocator, parent *version) (*version, error) {
|
2015-05-14 10:03:53 +00:00
|
|
|
re, err := regexp.Compile(`/vfs_([0-9a-f])$`)
|
2015-05-10 05:14:15 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
matches := re.FindStringSubmatch(base)
|
|
|
|
if len(matches) < 2 {
|
|
|
|
return nil, fmt.Errorf("invalid version identifier for %s", base)
|
|
|
|
}
|
|
|
|
|
|
|
|
timeval, err := strconv.ParseInt(matches[1], 16, 64)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2015-05-10 08:08:04 +00:00
|
|
|
ver := &version{
|
2015-05-10 07:00:35 +00:00
|
|
|
base: base,
|
|
|
|
parent: parent,
|
2015-05-16 07:50:15 +00:00
|
|
|
timestamp: time.Unix(timeval, 0),
|
|
|
|
inodeAloc: allocator}
|
2015-05-10 07:00:35 +00:00
|
|
|
|
2015-05-15 04:31:57 +00:00
|
|
|
ver.meta, err = newVersionMetadata(ver.metadataPath())
|
|
|
|
if err != nil {
|
2015-05-10 08:08:04 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return ver, nil
|
2015-05-10 07:00:35 +00:00
|
|
|
}
|
|
|
|
|
2015-05-16 07:43:46 +00:00
|
|
|
func (this *version) newVersionedNode(path string, info os.FileInfo) *versionedNode {
|
|
|
|
return &versionedNode{path, info, this}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (this *version) scanDir(path string) (versionedNodeMap, error) {
|
|
|
|
var baseNodes versionedNodeMap
|
2015-05-14 05:12:23 +00:00
|
|
|
if this.parent != nil {
|
|
|
|
var err error
|
2015-05-15 04:15:19 +00:00
|
|
|
|
2015-05-14 05:12:23 +00:00
|
|
|
baseNodes, err = this.parent.scanDir(path)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2015-05-15 04:15:19 +00:00
|
|
|
|
2015-05-16 07:07:45 +00:00
|
|
|
this.meta.filter(baseNodes)
|
2015-05-14 05:12:23 +00:00
|
|
|
}
|
|
|
|
|
2015-05-16 07:43:46 +00:00
|
|
|
ownNodes := make(versionedNodeMap)
|
2015-05-12 08:20:13 +00:00
|
|
|
{
|
2015-05-14 11:24:11 +00:00
|
|
|
nodes, err := ioutil.ReadDir(this.rebasePath(path))
|
2015-05-14 12:00:14 +00:00
|
|
|
if !os.IsNotExist(err) {
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2015-05-12 08:20:13 +00:00
|
|
|
|
2015-05-14 12:00:14 +00:00
|
|
|
for _, node := range nodes {
|
2015-05-16 07:43:46 +00:00
|
|
|
ownNodes[node.Name()] = this.newVersionedNode(filepath.Join(path, node.Name()), node)
|
2015-05-14 12:00:14 +00:00
|
|
|
}
|
2015-05-12 08:20:13 +00:00
|
|
|
}
|
2015-05-15 04:15:19 +00:00
|
|
|
|
2015-05-16 07:07:45 +00:00
|
|
|
this.meta.filter(ownNodes)
|
2015-05-12 08:20:13 +00:00
|
|
|
}
|
|
|
|
|
2015-05-14 05:12:23 +00:00
|
|
|
if baseNodes == nil {
|
2015-05-12 08:20:13 +00:00
|
|
|
return ownNodes, nil
|
|
|
|
}
|
2015-05-10 10:41:15 +00:00
|
|
|
|
2015-05-14 05:12:23 +00:00
|
|
|
for ownName, ownNode := range ownNodes {
|
|
|
|
baseNodes[ownName] = ownNode
|
2015-05-12 08:20:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return baseNodes, nil
|
2015-05-10 10:41:15 +00:00
|
|
|
}
|
|
|
|
|
2015-05-16 07:12:39 +00:00
|
|
|
func (this *version) buildVerDir(dir *versionedDir) error {
|
|
|
|
nodes, err := this.scanDir(dir.node.path)
|
2015-05-12 08:20:13 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2015-05-13 11:11:32 +00:00
|
|
|
for name, node := range nodes {
|
|
|
|
if node.info.IsDir() {
|
2015-05-18 01:23:53 +00:00
|
|
|
subDir := newVersionedDir(node, this.inodeAloc.AllocInode(), dir)
|
2015-05-16 07:12:39 +00:00
|
|
|
if err := this.buildVerDir(subDir); err != nil {
|
2015-05-12 09:02:27 +00:00
|
|
|
return err
|
|
|
|
}
|
2015-05-14 05:12:23 +00:00
|
|
|
|
2015-05-12 09:02:27 +00:00
|
|
|
dir.dirs[name] = subDir
|
2015-05-12 08:20:13 +00:00
|
|
|
} else {
|
2015-05-18 01:23:53 +00:00
|
|
|
dir.files[name] = newVersionedFile(node, this.inodeAloc.AllocInode(), dir)
|
2015-05-12 08:20:13 +00:00
|
|
|
}
|
2015-05-10 10:41:15 +00:00
|
|
|
}
|
|
|
|
|
2015-05-12 08:20:13 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-05-14 12:00:14 +00:00
|
|
|
func (this *version) resolve() error {
|
2015-05-14 11:24:11 +00:00
|
|
|
node, err := os.Stat(this.rebasePath("/"))
|
2015-05-12 09:02:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2015-05-14 05:24:52 +00:00
|
|
|
this.root = newVersionedDir(
|
2015-05-16 07:43:46 +00:00
|
|
|
this.newVersionedNode("/", node),
|
2015-05-18 01:23:53 +00:00
|
|
|
this.inodeAloc.AllocInode(),
|
|
|
|
nil)
|
2015-05-13 11:11:32 +00:00
|
|
|
|
2015-05-16 07:12:39 +00:00
|
|
|
return this.buildVerDir(this.root)
|
2015-05-10 10:41:15 +00:00
|
|
|
}
|
|
|
|
|
2015-05-10 07:15:17 +00:00
|
|
|
func (this *version) metadataPath() string {
|
2015-05-10 07:00:35 +00:00
|
|
|
return filepath.Join(this.base, "meta.json")
|
2015-05-08 09:01:24 +00:00
|
|
|
}
|
2015-05-10 08:08:04 +00:00
|
|
|
|
2015-05-14 11:24:11 +00:00
|
|
|
func (this *version) rebasePath(paths ...string) string {
|
|
|
|
combined := append([]string{this.base, "root"}, paths...)
|
|
|
|
return filepath.Join(combined...)
|
2015-05-10 08:08:04 +00:00
|
|
|
}
|
2015-05-12 08:20:13 +00:00
|
|
|
|
2015-05-12 10:48:11 +00:00
|
|
|
func (this *version) Root() (fs.Node, error) {
|
|
|
|
return this.root, nil
|
|
|
|
}
|
2015-05-14 06:10:42 +00:00
|
|
|
|
|
|
|
func (this *version) dump(root *versionedDir, depth int) {
|
|
|
|
indent := strings.Repeat("\t", depth)
|
|
|
|
for name, dir := range root.dirs {
|
2015-05-16 07:07:45 +00:00
|
|
|
fmt.Printf("%s+ %s [%s@%d]\n", indent, name, dir.node.path, dir.node.ver.timestamp.Unix())
|
2015-05-14 06:10:42 +00:00
|
|
|
this.dump(dir, depth+1)
|
|
|
|
}
|
|
|
|
for name, file := range root.files {
|
2015-05-16 07:07:45 +00:00
|
|
|
fmt.Printf("%s- %s [%s@%d]\n", indent, name, file.node.path, file.node.ver.timestamp.Unix())
|
2015-05-14 06:10:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (this *version) dumpRoot() {
|
|
|
|
this.dump(this.root, 0)
|
|
|
|
}
|