package master

import (
    "net/http"
    "io/ioutil"
    "encoding/json"
    "time"
    "strings"
    "sync"
    "math/rand"
    "path/filepath"
    "fmt"
    "os"
    "github.com/030io/whalefs/utils/uuid"
)

type Size interface {
    Size() int64
}
//公共资源访问处理器
func (m *Master)publicEntry(w http.ResponseWriter, r *http.Request) {
    m.serverMutex.RLock()
    defer m.serverMutex.RUnlock()

    if r.Method == http.MethodGet || r.Method == http.MethodHead {
        m.getFile(w, r)
    } else {
        w.WriteHeader(http.StatusMethodNotAllowed)
    }
}
//卷文件访问
func (m *Master)masterEntry(w http.ResponseWriter, r *http.Request) {
    m.serverMutex.RLock()
    defer m.serverMutex.RUnlock()

    switch r.URL.Path {
    case "/__heartbeat":
        m.heartbeat(w, r)
    default:
        if r.URL.Path == "/favicon.ico" || len(r.URL.Path) <= 1 {
            http.NotFound(w, r)
            return
        }

        switch r.Method{
        case http.MethodGet, http.MethodHead:
            m.getFile(w, r)
        case http.MethodPost:
            m.uploadFile(w, r)
        case http.MethodDelete:
            m.deleteFile(w, r)
        default:
            w.WriteHeader(http.StatusMethodNotAllowed)
        }
    }
}
//心跳连接处理器
func (m *Master)heartbeat(w http.ResponseWriter, r *http.Request) {
    body, _ := ioutil.ReadAll(r.Body)
    newVms := new(VolumeManagerStatus)
    json.Unmarshal(body, newVms)
    newVms.LastHeartbeat = time.Now()

    remoteIP := r.RemoteAddr[:strings.LastIndex(r.RemoteAddr, ":")]
    if newVms.AdminHost == "" || newVms.AdminHost == "localhost" {
        newVms.AdminHost = remoteIP
    }
    if newVms.PublicHost == "" || newVms.PublicHost == "localhost" {
        newVms.PublicHost = remoteIP
    }
    if newVms.Machine == "" {
        newVms.Machine = remoteIP
    }

    m.updateVMS(newVms)

    if m.vmsNeedCreateVolume(newVms) {
        go m.createVolumeWithReplication(newVms)
    }
}

func (m *Master)getFile(w http.ResponseWriter, r *http.Request) {
    vid, fid, fileName, err := m.Metadata.Get(r.URL.Path)
    if err != nil {
        http.NotFound(w, r)
        return
    }

    m.statusMutex.RLock()
    vStatusList, ok := m.VStatusListMap[vid]
    m.statusMutex.RUnlock()
    if !ok {
        http.Error(w, "can't find volume", http.StatusNotFound)
        return
    }

    length := len(vStatusList)
    j := rand.Intn(length)
    for i := 0; i < length; i++ {
        vStatus := vStatusList[(i + j) % length]
        if vStatus.vmStatus.IsAlive() {
            http.Redirect(w, r, vStatus.getFileUrl(fid, fileName), http.StatusFound)
            return
        }
    }

    http.Error(w, "all volumes is dead", http.StatusInternalServerError)
}

func (m *Master)uploadFile(w http.ResponseWriter, r *http.Request) {
    file, header, err := r.FormFile("file")
    if err != nil {
        http.Error(w, "r.FromFile: " + err.Error(), http.StatusInternalServerError)
        return
    }
    defer file.Close()

    var dst string
    if r.URL.Path[len(r.URL.Path) - 1] == '/' {
        dst = r.URL.Path + filepath.Base(header.Filename)
    } else {
        dst = r.URL.Path
    }
    fileName := filepath.Base(dst)

    if m.Metadata.Has(dst) {
        http.Error(w, "file is existed, you should delete it at first.", http.StatusNotAcceptable)
        return
    }

    var fileSize int64
    switch file.(type){
    case *os.File:
        s, _ := file.(*os.File).Stat()
        fileSize = s.Size()
    case Size:
        fileSize = file.(Size).Size()
    }

    vStatusList, err := m.getWritableVolumes(uint64(fileSize))
    if err != nil {
        http.Error(w, "m.getWritableVolumes: " + err.Error(), http.StatusInternalServerError)
        return
    }

    data, _ := ioutil.ReadAll(file)
    fid := uuid.GenerateUUID()

    wg := sync.WaitGroup{}
    var errVS *VolumeStatus
    for _, vStatus := range vStatusList {
        wg.Add(1)
        go func(vs *VolumeStatus) {
            e := vs.uploadFile(fid, fileName, data)
            if e != nil {
                err = e
                errVS = vs
            }
            wg.Done()
        }(vStatus)
    }
    wg.Wait()

    if err != nil {
        for _, vStatus := range vStatusList {
            go vStatus.delete(fid, fileName)
        }
        http.Error(w,
            fmt.Sprintf("host: %s port: %d error: %s", errVS.vmStatus.AdminHost, errVS.vmStatus.AdminPort, err),
            http.StatusInternalServerError)
        return
    }

    m.Metadata.Set(dst, vStatusList[0].Id, fid, fileName)
    if err != nil {
        http.Error(w, "m.Metadata.Set: " + err.Error(), http.StatusInternalServerError)
        return
    }
    w.WriteHeader(http.StatusCreated)
}

func (m *Master)deleteFile(w http.ResponseWriter, r *http.Request) {
    vid, fid, fileName, err := m.Metadata.Get(r.URL.Path)
    if err != nil {
        http.NotFound(w, r)
        return
    }

    m.statusMutex.RLock()
    vStatusList, ok := m.VStatusListMap[vid]
    m.statusMutex.RUnlock()
    if !ok {
        http.Error(w, "can't find volume", http.StatusNotFound)
        return
    } else if !m.volumesIsValid(vStatusList) || !volumesIsWritable(vStatusList, 0) {
        http.Error(w, "can't delete file, because it's(volumes) readonly.", http.StatusNotAcceptable)
    }

    wg := sync.WaitGroup{}
    var deleteErr []error
    for _, vStatus := range vStatusList {
        wg.Add(1)
        go func(vStatus *VolumeStatus) {
            e := vStatus.delete(fid, fileName)
            if e != nil {
                deleteErr = append(
                    deleteErr,
                    fmt.Errorf("%s:%d %s", vStatus.vmStatus.AdminHost, vStatus.vmStatus.AdminPort, e.Error()),
                )
            }
            wg.Done()
        }(vStatus)
    }
    wg.Wait()

    err = m.Metadata.Delete(r.URL.Path)
    if err != nil {
        deleteErr = append(deleteErr, fmt.Errorf("m.Metadata.Delete(%s) %s", r.URL.Path, err.Error()))
    }

    if len(deleteErr) == 0 {
        w.WriteHeader(http.StatusAccepted)
    } else {
        errStr := ""
        for _, err := range deleteErr {
            errStr += err.Error() + "\n"
        }
        http.Error(w, errStr, http.StatusInternalServerError)
        return
    }
}

handler.go的更多相关文章

  1. android Handler介绍

    Handler使用介绍: Handler根据接收的消息,处理UI更新.Thread线程发出消息,通知Handler更新UI. Handler mHandler = new Handler() {  p ...

  2. Handler

    1.1 继承AbstractController优点:能定制请求方式 package cn.happyl.controller; import javax.servlet.http.HttpServl ...

  3. Android消息处理机制(Handler、Looper、MessageQueue与Message)

    Android是消息驱动的,实现消息驱动有几个要素: 消息的表示:Message 消息队列:MessageQueue 消息循环,用于循环取出消息进行处理:Looper 消息处理,消息循环从消息队列中取 ...

  4. Android笔记——Handler Runnable与Thread的区别

    在java中可有两种方式实现多线程,一种是继承Thread类,一种是实现Runnable接口:Thread类是在java.lang包中定义的.一个类只要继承了Thread类同时覆写了本类中的run() ...

  5. Android消息传递之Handler消息机制

    前言: 无论是现在所做的项目还是以前的项目中,都会遇见线程之间通信.组件之间通信,目前统一采用EventBus来做处理,在总结学习EventBus之前,觉得还是需要学习总结一下最初的实现方式,也算是不 ...

  6. Handler系列之内存泄漏

    本篇简单的讲一下平常使用Handler时造成内存泄漏的问题. 什么是内存泄漏?大白话讲就是分配出去的内存,回收不回来.严重会导致内存不足OOM.下面来看一下造成内存泄漏的代码: public clas ...

  7. Handler系列之创建子线程Handler

    上一篇我介绍了Handler机制的工作原理,默认情况下,ActivityThread类为我们创建的了主线程的Looper和消息队列,所以当你创建Handler之后发送消息的时候,消息的轮训和handl ...

  8. Handler系列之原理分析

    上一节我们讲解了Handler的基本使用方法,也是平时大家用到的最多的使用方式.那么本节让我们来学习一下Handler的工作原理吧!!! 我们知道Android中我们只能在ui线程(主线程)更新ui信 ...

  9. Handler系列之使用

    作为一个Android开发者,我们肯定熟悉并使用过Handler机制.最常用的使用场景是"在子线程更新ui",实际上我们知道上面的说话是错误的.因为Android中只有主线程才能更 ...

  10. 阶段一:用Handler和Message实现计时效果及其中一些疑问

    “阶段一”是指我第一次系统地学习Android开发.这主要是对我的学习过程作个记录. 本来是打算继续做天气预报的优化的,但因为某些原因,我要先把之前做的小应用优化一下.所以今天就插播一下用Handle ...

随机推荐

  1. LeetCode(29)-Plus One

    题目: Given a non-negative number represented as an array of digits, plus one to the number. The digit ...

  2. python简单线程和协程学习

    python中对线程的支持的确不够,不过据说python有足够完备的异步网络框架模块,希望日后能学习到,这里就简单的对python中的线程做个总结 threading库可用来在单独的线程中执行任意的p ...

  3. (WPS) 网络地理信息处理服务

    WPS 标准为网络地理信息处理服务提供了标准化的输入和输出. OGC® Web Processing Service (WPS) 标准描述了如何通过远程的任何算法和模型处理获得地理空间的栅格或矢量信息 ...

  4. IoC和DI的基本概念的思维导图

    最近在学习Spring开发,IoC这个概念让我有点儿迷糊,控制反转这四个字是在是无法做到望文生义,于是乎就找了一些材料来学习,研究了半天,绘制了下面这幅思维导图.仅供参考!

  5. w3school上系统过了一遍Jquery的总结

    下面是今天学习JQUERY中发现的一些小钻石 1:$(document).ready(function(){}); 为了防止文档在完全加载(就绪)之前运行 jQuery 代码(终于搞清了这句代码的含义 ...

  6. MySQL/MariaDB的锁

    本文目录: 1.MariaDB/MySQL事务提交的方式 2.MariaDB/MySQL中的锁简介 2.1 不同存储引擎支持的锁级别 2.2 锁类型 2.3 锁兼容性 3.MyISAM的表级锁(loc ...

  7. InnoDB的4个特性

    innodb 的四个特性 insert buffer innodb使用insert buffer"欺骗"数据库:对于为非唯一索引,辅助索引的修改操作并非实时更新索引的叶子页,而是把 ...

  8. Web开发问题记录

    1.先说一个CSS的:CSS中带有中文(比如字体定义)的属性定义最好放在该选择器定义诸项的最后一条,为什么----编码格式问题. 2.其实自己也可以用自己写的DispatcherServlet+jsp ...

  9. Ubuntu14.04部署pyspider的过程

    1.安装,安装官方文档,应该先执行 sudo apt-get install python python-dev python-distribute python-pip libcurl4-opens ...

  10. Android版本分布数据源

    先来Android官方数据地址:http://developer.android.com/intl/zh-cn/about/dashboards/index.html 友盟指数,这个对国内开发者比较有 ...