package volume

import (
    "os"
    "path/filepath"
    "strconv"
    "sync"
    "time"
    "encoding/binary"
    "errors"
)

var (
    TruncateSize uint64 = 1 << 30 //1GB
    MaxVolumeSize uint64 = 512 * TruncateSize //512GB
)

type Volume struct {
    Id        uint64
    WriteAble bool
    index     Index
    status    *Status
    dataFile  *os.File
    mutex     sync.Mutex
}
//创建一个卷 或者磁盘空间   
func NewVolume(dir string, vid uint64) (v *Volume, err error) {
    path := filepath.Join(dir, strconv.FormatUint(vid, 10) + ".data")
    v = new(Volume)
    v.Id = vid
    v.dataFile, err = os.OpenFile(path, os.O_CREATE | os.O_RDWR, 0666)
    if err != nil {
        if os.IsPermission(err) {//判断是不是权限错误
            v.dataFile, err = os.OpenFile(path, os.O_RDONLY, 0)
            if err != nil {
                return nil, err
            }
            v.WriteAble = false
        } else {
            return nil, err
        }
    }
    v.WriteAble = true

    v.index, err = NewLevelDBIndex(dir, vid)
    if err != nil {
        return nil, err
    }

    v.status, err = NewStatus(dir, vid)
    if err != nil {
        return nil, err
    }

    return v, nil
}
//实现index接口
//
func (v *Volume)Get(fid uint64) (*File, error) {
    fi, err := v.index.Get(fid)
    if err == nil {
        return &File{DataFile: v.dataFile, Info: fi}, nil
    } else {
        return nil, err
    }
}
//删除文件
func (v *Volume)Delete(fid uint64, fileName string) error {
    v.mutex.Lock()
    defer v.mutex.Unlock()
    file, err := v.Get(fid)
    if err != nil {
        return err
    } else if file.Info.FileName != fileName {
        return errors.New("filename is wrong")
    }
    //因为文件内容前后都写入fid(8 byte) 要一起释放
    err = v.status.freeSpace(file.Info.Offset - 8, file.Info.Size + 16)
    if err != nil {
        return err
    }
    err = v.index.Delete(fid)
    return err
}
//创建文件
func (v *Volume)NewFile(fid uint64, fileName string, size uint64) (f *File, err error) {
    v.mutex.Lock()
    defer v.mutex.Unlock()
    if v.index.Has(fid) {
        return nil, errors.New("fid is existed")
    }
    offset, err := v.newSpace(size + 16)
    if err != nil {
        return nil, err
    }
    defer func() {
        if err != nil {
            if e := v.status.freeSpace(offset, size + 16); e != nil {
                panic(e)
            }
        }
    }()
    //在文件内容前后都写入fid
    b := make([]byte, 8)
    binary.BigEndian.PutUint64(b, fid)
    _, err = v.dataFile.WriteAt(b, int64(offset))
    if err != nil {
        return nil, err
    }
    _, err = v.dataFile.WriteAt(b, int64(offset + 8 + size))
    if err != nil {
        return nil, err
    }
    fileInfo := &FileInfo{
        Fid: fid,
        Offset: offset + 8,
        Size: size,
        Ctime: time.Now(),
        Mtime: time.Now(),
        Atime: time.Now(),
        FileName: fileName,
    }
    err = v.index.Set(fileInfo)
    if err != nil {
        return nil, err
    } else {
        return &File{DataFile: v.dataFile, Info: fileInfo}, nil
    }
}
//清空文件空间
func (v *Volume)truncate() {
    currentDatafileSize := v.GetDatafileSize()
    if currentDatafileSize >= MaxVolumeSize {
        return
    }
       //清空指定偏移量位置的数据
    err := v.dataFile.Truncate(int64(currentDatafileSize) + int64(TruncateSize))
    if err != nil {
        panic(err)
    }
       //释放文件空间
    err = v.status.freeSpace(currentDatafileSize, TruncateSize)
    if err != nil {
        panic(err)
    }
}
//分配新的文件空间
func (v *Volume)newSpace(size uint64) (uint64, error) {
    offset, err := v.status.newSpace(size)
    if err == nil {
        return offset, err
    }
    v.truncate()
    return v.status.newSpace(size)
}
//资源释放
func (v *Volume)Close() {
    v.mutex.Lock()
    //v.status.spaceMutex.Lock()
    //因为要退出,所以不解锁,禁止读写
    //defer v.rwMutex.Unlock()

    //将所有资源释放
    v.dataFile.Close()
    v.dataFile = nil

    v.status.db.Close()
    v.status = nil

    v.index.Close()
    v.index = nil
}
//获取数据文件大小
func (v *Volume)GetDatafileSize() uint64 {
    fi, err := v.dataFile.Stat()
    if err != nil {
        panic(err)
    }
    return uint64(fi.Size())
}
//获取最大释放文件空间
func (v *Volume)GetMaxFreeSpace() uint64 {
    currentDatafileSize := v.GetDatafileSize()
    //以块方式存储,所以自由空间无法直接相加
    maxFreeSpace := v.status.getMaxFreeSpace()
    if currentDatafileSize < MaxVolumeSize {
        freeSpace := MaxVolumeSize - currentDatafileSize
        if freeSpace > maxFreeSpace {
            maxFreeSpace = freeSpace
        }
    }
    //fid前后各占8个字节
    if maxFreeSpace > 16 {
        return maxFreeSpace - 16
    } else {
        return 0
    }
}

volume.go的更多相关文章

  1. Java中实现SAX解析xml文件到MySQL数据库

    大致步骤: 1.Java bean 2.DBHelper.java 3.重写DefaultHandler中的方法:MyHander.java 4.循环写数据库:SAXParserDemo.java ① ...

  2. CSharpGL(8)使用3D纹理渲染体数据 (Volume Rendering) 初探

    CSharpGL(8)使用3D纹理渲染体数据 (Volume Rendering) 初探 2016-08-13 由于CSharpGL一直在更新,现在这个教程已经不适用最新的代码了.CSharpGL源码 ...

  3. 理解Docker(8):Docker 存储之卷(Volume)

    (1)Docker 安装及基本用法 (2)Docker 镜像 (3)Docker 容器的隔离性 - 使用 Linux namespace 隔离容器的运行环境 (4)Docker 容器的隔离性 - 使用 ...

  4. Docker Volume 之权限管理(转)

    Volume数据卷是Docker的一个重要概念.数据卷是可供一个或多个容器使用的特殊目录,可以为容器应用存储提供有价值的特性: 持久化数据与容器的生命周期解耦:在容器删除之后数据卷中的内容可以保持.D ...

  5. NFS Volume Provider(Part III) - 每天5分钟玩转 OpenStack(64)

    今天我们将前一小节创建的 NFS volume “nfs-vol-1” attach 到 instance “c2”上. 这里我们重点关注 nova-compute 如何将“nfs-vol-1” at ...

  6. NFS Volume Provider(Part II) - 每天5分钟玩转 OpenStack(63)

    上一节我们将 NFS volume provider 配置就绪,本节将创建 volume. 创建 volume 创建 NFS volume 操作方法与 LVM volume 一样,唯一区别是在 vol ...

  7. NFS Volume Provider(Part I) - 每天5分钟玩转 OpenStack(62)

    cinder-volume 支持多种 volume provider,前面我们一直使用的是默认的 LVM,本节我们将增加 NFS volume provider. 虽然 NFS 更多地应用在实验或小规 ...

  8. Boot from Volume - 每天5分钟玩转 OpenStack(61)

    Volume 除了可以用作 instance 的数据盘,也可以作为启动盘(Bootable Volume),那么如何使 volume 成为 bootable 呢? 现在我们打开 instance 的 ...

  9. Restore Volume 操作 - 每天5分钟玩转 OpenStack(60)

    前面我们 backup 了 voluem,今天我们将讨论如何 restore volume. restore 的过程其实很简单,两步走: 在存储节点上创建一个空白 volume. 将 backup 的 ...

  10. Backup Volume 操作 - 每天5分钟玩转 OpenStack(59)

    本节我们讨论 volume 的 Backup 操作. Backup 是将 volume 备份到别的地方(备份设备),将来可以通过 restore 操作恢复. Backup VS Snapshot 初看 ...

随机推荐

  1. 【模板】最近公共祖先(LCA)

    题目描述 如题,给定一棵有根多叉树,请求出指定两个点直接最近的公共祖先. 输入输出格式 输入格式: 第一行包含三个正整数N.M.S,分别表示树的结点个数.询问的个数和树根结点的序号. 接下来N-1行每 ...

  2. MySQL 表名区分大小写设置

    1.关闭MySQL服务:         控制面板主页-管理工具-服务-MySQL服务 2.在服务器运行目录找到my.ini 或者my.cnf文件: 在[mysqld]下面增加一行添加 :lower_ ...

  3. 清楚css浮动的三种方法

    第一种:添加新元素,应用clear:both;  <div class="clear"></div> css样式:clear:both; 第二种:在浮动元素 ...

  4. 一个SQL存储过程面试题(比较简单)

    三个月前刚毕业的时候,听到存储过程就头疼. 写一个SQL存储过程,建立一个表USER 字段是姓名,年龄,职位,权限,然后向里面插入6条数据,然后查询出年龄大于18的所有信息. 下面是答案: 复制代码 ...

  5. LAMP的搭建

    可以在网上找整合包: httpd-2.4.18 + mysql-5.6.29 + php-5.5.30编译安装过程: 编译源代码后安装软件的位置:/usr/local/ 一.编译安装apache ap ...

  6. codechef Killing Monsters

    题目大意:大厨正在玩一个打怪兽的小游戏.游戏中初始时有 n 只怪兽排成一排,从左到右编号为 0 ∼ n − 1.第 i 只怪兽的初始血量为 hi,当怪兽的血量小于等于 0 时,这只怪兽就挂了. 大厨要 ...

  7. mac上Python多版本共存

    http://www.cnblogs.com/mingaixin/p/6295963.html https://www.cnhzz.com/pyenv_virtualenv_virtaulenvwra ...

  8. c#之多线程之为所欲为

    一 什么是多线程 1. 什么是进程?一个 exe 运行一次就会产生一个进程,一个 exe 的多个进程之 间数据互相隔离. 2. 一个进程里至少有一个线程:主线程.我们平时写的控制台程序默认就是单线程的 ...

  9. Vue、AngularJS 双向数据绑定解剖

    数据与视图的绑定与同步,最终体现在对数据的读写处理过程中,也就是 Object.defineProperty() 定义的数据 set.get 函数中.Vue 中对于的函数为 defineReactiv ...

  10. c#学习笔记 day_one

    C#学习笔记 day one Chapter 1 c#概述 1.1 c#概述 C#是微软设计的,简洁的,类型安全的,面向对象的语言.它以c/c++作为基础.它的开发环境是visual studio,最 ...