package manager

import (
    "net/http"
    "github.com/030io/whalefs/manager/volume"
    "os"
    "io/ioutil"
    "strings"
    "strconv"
    "fmt"
    "time"
    "github.com/030io/whalefs/master/api"
    "github.com/030io/whalefs/master"
    "github.com/030io/whalefs/utils/disk"
)

var (
    MaxDiskUsedPercent uint = 99
    HeartbeatDuration time.Duration = time.Second * 5
    ReadOnly bool = false
    DefaultExpires = time.Minute * 30
)

type VolumeManager struct {
    DataDir      string
    Volumes      map[uint64]*volume.Volume
    AdminPort    int
    AdminHost    string
    PublicPort   int
    PublicHost   string
    AdminServer  *http.ServeMux
    PublicServer *http.ServeMux
    Machine      string
    DataCenter   string
    MasterHost   string
    MasterPort   int
}

func NewVolumeManager(dir string) (*VolumeManager, error) {
    f, err := os.OpenFile(dir, os.O_RDWR, 0)
    if os.IsNotExist(err) {
        panic(err)
    } else if os.IsPermission(err) {
        ReadOnly = true
    }
    f.Close()
    vm := new(VolumeManager)
    vm.DataDir = dir
    fileInfos, err := ioutil.ReadDir(dir)
    if err != nil {
        panic(err)
    }
    vm.Volumes = make(map[uint64]*volume.Volume)
    for _, fi := range fileInfos {
        fileName := fi.Name()
        if strings.HasSuffix(fileName, ".data") {
            vid, err := strconv.ParseUint(fileName[:len(fileName) - 5], 10, 64)
            if err != nil {
                panic(err)
            }
            vm.Volumes[vid], err = volume.NewVolume(dir, vid)
            if err != nil {
                panic(err)
            }
        }
    }
    vm.AdminPort = 7800
    vm.AdminHost = "localhost"
    vm.PublicPort = 7900
    vm.PublicHost = "localhost"
    vm.AdminServer = http.NewServeMux()
    vm.PublicServer = http.NewServeMux()
    vm.PublicServer.HandleFunc("/", vm.publicEntry)
    vm.AdminServer.HandleFunc("/", vm.adminEntry)
    vm.MasterHost = "localhost"
    vm.MasterPort = 8888
    return vm, nil
}

func (vm *VolumeManager)Start() {
    go vm.Heartbeat()

    go func() {
        err := http.ListenAndServe(fmt.Sprintf("0.0.0.0:%d", vm.AdminPort), vm.AdminServer)
        if err != nil {
            panic(err)
        }
    }()

    err := http.ListenAndServe(fmt.Sprintf("0.0.0.0:%d", vm.PublicPort), vm.PublicServer)
    if err != nil {
        panic(err)
    }
}

func (vm *VolumeManager)Stop() {
    for _, v := range vm.Volumes {
        v.Close()
    }
}

func (vm *VolumeManager)Heartbeat() {
    tick := time.NewTicker(HeartbeatDuration)
    defer tick.Stop()
    for {
        vms := new(master.VolumeManagerStatus)
        vms.AdminHost = vm.AdminHost
        vms.AdminPort = vm.AdminPort
        vms.PublicHost = vm.PublicHost
        vms.PublicPort = vm.PublicPort
        vms.Machine = vm.Machine
        vms.DataCenter = vm.Machine
        vms.VStatusList = make([]*master.VolumeStatus, 0, len(vm.Volumes))

        diskUsage, _ := disk.DiskUsage(vm.DataDir)
        vms.DiskSize = diskUsage.Size
        vms.DiskUsed = diskUsage.Used
        vms.DiskFree = diskUsage.Free
        vms.MaxDiskUsed = diskUsage.Size / 100 * uint64(MaxDiskUsedPercent)
        vms.VolumeMaxSize = volume.MaxVolumeSize

        diskUsedPercent := uint(float64(diskUsage.Used) / float64(diskUsage.Size) * 100)
        if ReadOnly || diskUsedPercent >= MaxDiskUsedPercent {
            //禁止所有volume再进行truncate
            volume.MaxVolumeSize = 0
            vms.CanCreateVolume = false
        } else {
            vms.CanCreateVolume = true
        }

        for vid, v := range vm.Volumes {
            vs := new(master.VolumeStatus)
            vs.Id = vid
            vs.DataFileSize = v.GetDatafileSize()
            vs.Writable = !ReadOnly && v.WriteAble
            vs.MaxFreeSpace = v.GetMaxFreeSpace()
            vms.VStatusList = append(vms.VStatusList, vs)
        }

        api.Heartbeat(vm.MasterHost, vm.MasterPort, vms)
        <-tick.C
    }
}

volume_manager.go的更多相关文章

  1. Kubeadm安装Kubernetes环境

    Kubeadm方式号称一键安装部署,很多人也试过并且顺利成功,可到了我这里因为折腾系统问题,倒腾出不少的坑出来. kubeadm好处是自动配置了必要的服务,以及缺省配置了安全的认证,etcd,apis ...

  2. 基于openstack stable queens版本阅读解析

    基于openstack stable queens版本阅读解析 基于 centos7.5 的linux系统 架构 如下所示,为cinder的官方架构说明: 这里写图片描述 各个组件介绍如下: - DB ...

  3. cinder-volume服务上报自己的状态给cinder-scheduler的rpc通信代码分析

    以juno版本为基础,主要从消息的生产者-消费者模型及rpc client/server模型来分析cinder-volume是如何跟cinder-scheduler服务进行rpc通信的 1.cinde ...

  4. kubelet之volume manager源码分析

    kubernetes ceph-csi分析目录导航 基于tag v1.17.4 https://github.com/kubernetes/kubernetes/releases/tag/v1.17. ...

随机推荐

  1. java中split(regex)使用中要注意的问题:正则表达式

    比如我在项目中遇到的(,),.,|,*等等类的符号: String area="(30.13206313822174, 120.4156494140625)(29.8763738070713 ...

  2. mac os x下Dreamweaver如何还原初始配置

    上次在mac下修改Dreamweaver(以下简称dw)时,不知动了哪里,导致打开html文档时设计按钮变灰不能使用!这个太蛋疼了,只能在浏览器中查看效果,live按钮更不用说也是灰化状态. 于是使用 ...

  3. jQuery结合lhgdialog弹出窗口,关闭时出现没有权限错误

    背景: 最近的项目,使用JQuery+lhgdialog窗口组件方式模拟弹窗,在关闭lhgdialog窗口时,出现以下错误: >jQuery没有权限 >调试时 w.readyState没有 ...

  4. Object对象你真理解了吗?

    前言 五一回家又断更了一个放假时间了~~~ 只有光头才能变强 回顾前面: ThreadLocal就是这么简单 多线程三分钟就可以入个门了! 多线程基础必要知识点!看了学习多线程事半功倍 Java锁机制 ...

  5. Emmet之html语法

    一.简写语法 Emmet 用和 CSS 选择器相似的语法来描述元素的嵌套层级关系和属性,实现 HTML/XML/CSS 等代码的智能自动补全. 其通过文件名后缀识别文件类型,从而使用对应的自动补全语法 ...

  6. jdk1.7 tomcat-7安装

    由于软件下载地址经常有变动,所以不能直接wget,还是直接到网上点击下载 下载jdk http://www.oracle.com/technetwork/java/javase/downloads/j ...

  7. 北京一家JAVA开发公司面试题(留给后人)

    1.jsp有哪些内置对象?作用分别是什么? 2.描述一下servlet的生命周期和基本架构. 3.多线程有几种实现方法,都是什么? 同步有几种实现方法,都是什么? 4.作用域public   priv ...

  8. Android框架之Volley与Glide

    PS:在看到这个题目的同时,你们估计会想,Volley与Glide怎么拿来一块说呢,他们虽然不是一个框架,但有着相同功能,那就是图片处理方面.首先我们先来看一下什么volley,又什么是glide. ...

  9. Spring消息之STOMP

    一.STOMP 简介 直接使用WebSocket(或SockJS)就很类似于使用TCP套接字来编写Web应用.因为没有高层级的线路协议(wire protocol),因此就需要我们定义应用之间所发送消 ...

  10. angular2 安装 打包成发布项目过程

    安装之前要有typings和typescript全局已经安装好 安装命令新版为npm install -g @angular/cli 原来的angular-cli为老版的,我安装失败了 安装之后新建一 ...