Merge pull request #724 from harshavardhana/pr_out_add_mountinfo_functions_for_detecting_mount_disks

master
Harshavardhana 10 years ago
commit 4f29dc9134
  1. 2
      commands.go
  2. 4
      pkg/controller/client.go
  3. 11
      pkg/server/rpc/diskinfo.go
  4. 122
      pkg/utils/scsi/mountinfo.go
  5. 5
      pkg/utils/scsi/scsi_test.go

@ -110,7 +110,7 @@ func runController(c *cli.Context) {
if err != nil { if err != nil {
Fatalln(err) Fatalln(err)
} }
Println(disks) Println(string(disks))
case "mem": case "mem":
memstats, err := controller.GetMemStats(c.Args().Tail().First()) memstats, err := controller.GetMemStats(c.Args().Tail().First())
if err != nil { if err != nil {

@ -27,7 +27,7 @@ import (
) )
// GetDisks get disks info of the server at given url // GetDisks get disks info of the server at given url
func GetDisks(url string) ([]string, error) { func GetDisks(url string) ([]byte, error) {
op := RPCOps{ op := RPCOps{
Method: "DiskInfo.Get", Method: "DiskInfo.Get",
Request: rpc.Args{Request: ""}, Request: rpc.Args{Request: ""},
@ -45,7 +45,7 @@ func GetDisks(url string) ([]string, error) {
if err := jsonrpc.DecodeClientResponse(resp.Body, &reply); err != nil { if err := jsonrpc.DecodeClientResponse(resp.Body, &reply); err != nil {
return nil, iodine.New(err, nil) return nil, iodine.New(err, nil)
} }
return reply.Disks, nil return json.MarshalIndent(reply, "", "\t")
} }
// GetMemStats get memory status of the server at given url // GetMemStats get memory status of the server at given url

@ -30,8 +30,9 @@ type DiskInfoService struct{}
// DiskInfoReply disk info reply for disk info service // DiskInfoReply disk info reply for disk info service
type DiskInfoReply struct { type DiskInfoReply struct {
Hostname string `json:"hostname"` Hostname string `json:"hostname"`
Mounts map[string]scsi.Mountinfo `json:"mounts"`
Disks []string `json:"disks"` Disks []string `json:"disks"`
DiskAttributes map[string]scsi.Attributes `json:"disk-attrs"` DiskAttributes map[string]scsi.Attributes `json:"-"` // for the time being not unmarshalling this
} }
func setDiskInfoReply(sis *DiskInfoReply) error { func setDiskInfoReply(sis *DiskInfoReply) error {
@ -40,11 +41,19 @@ func setDiskInfoReply(sis *DiskInfoReply) error {
if err != nil { if err != nil {
return iodine.New(err, nil) return iodine.New(err, nil)
} }
mounts, err := scsi.GetMountInfo()
if err != nil {
return iodine.New(err, nil)
}
sis.Mounts = make(map[string]scsi.Mountinfo)
sis.Mounts = mounts
disks, err := scsi.GetDisks() disks, err := scsi.GetDisks()
if err != nil { if err != nil {
return iodine.New(err, nil) return iodine.New(err, nil)
} }
sis.DiskAttributes = make(map[string]scsi.Attributes) sis.DiskAttributes = make(map[string]scsi.Attributes)
for k, v := range disks { for k, v := range disks {
sis.Disks = append(sis.Disks, k) sis.Disks = append(sis.Disks, k)
sis.DiskAttributes[k] = v sis.DiskAttributes[k] = v

@ -0,0 +1,122 @@
// +build linux
/*
* Mini Object Storage, (C) 2014 Minio, Inc.
*
* 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 scsi
import (
"bufio"
"errors"
"io/ioutil"
"os"
"path/filepath"
"strconv"
"strings"
"syscall"
"github.com/minio/minio/pkg/iodine"
)
// Mountinfo container to capture /etc/mtab mount structure
type Mountinfo struct {
FSName string /* name of mounted filesystem */
Dir string /* filesystem path prefix */
Type string /* mount type (see mntent.h) */
Opts string /* mount options (see mntent.h) */
Freq int /* dump frequency in days */
Passno int /* pass number on parallel fsck */
}
var supportedFSType = map[string]bool{
"ext4": true,
"xfs": true,
"ext3": true,
"btrfs": true,
"tmpfs": true,
"nfs": true,
}
func isSupportedType(t string) bool {
_, ok := supportedFSType[t]
return ok
}
// GetMountInfo - get mount info map
func GetMountInfo() (map[string]Mountinfo, error) {
f, err := os.Open("/etc/mtab")
if err != nil {
return nil, iodine.New(err, nil)
}
mntEnt := make(map[string]Mountinfo)
scanner := bufio.NewScanner(f)
for scanner.Scan() {
mtabEnt := strings.Split(scanner.Text(), " ")
mntInfo := Mountinfo{}
if len(mtabEnt) == 6 {
var err error
if !isSupportedType(mtabEnt[2]) {
continue
}
mntInfo.FSName, err = filepath.EvalSymlinks(mtabEnt[0])
if err != nil {
continue
}
mntInfo.Dir = mtabEnt[1]
mntInfo.Type = mtabEnt[2]
mntInfo.Opts = mtabEnt[3]
mntInfo.Freq, err = strconv.Atoi(mtabEnt[4])
if err != nil {
continue
}
mntInfo.Passno, err = strconv.Atoi(mtabEnt[5])
if err != nil {
continue
}
mntEnt[mntInfo.FSName] = mntInfo
}
}
return mntEnt, nil
}
// IsUsable provides a comprehensive way of knowing if the provided mountPath is mounted and writable
func IsUsable(mountPath string) (bool, error) {
mntpoint, err := os.Stat(mountPath)
if err != nil {
return false, iodine.New(err, nil)
}
parent, err := os.Stat(filepath.Join(mountPath, ".."))
if err != nil {
return false, iodine.New(err, nil)
}
mntpointSt := mntpoint.Sys().(*syscall.Stat_t)
parentSt := parent.Sys().(*syscall.Stat_t)
if mntpointSt.Dev == parentSt.Dev {
return false, iodine.New(errors.New("not mounted"), nil)
}
testFile, err := ioutil.TempFile(mountPath, "writetest-")
if err != nil {
return false, iodine.New(err, nil)
}
testFileName := testFile.Name()
// close the file, to avoid leaky fd's
testFile.Close()
if err := os.Remove(testFileName); err != nil {
return false, iodine.New(err, nil)
}
return true, nil
}

@ -34,3 +34,8 @@ func (s *MySuite) TestSCSI(c *C) {
_, err := GetDisks() _, err := GetDisks()
c.Assert(err, IsNil) c.Assert(err, IsNil)
} }
func (s *MySuite) TestMountInfo(c *C) {
_, err := GetMountInfo()
c.Assert(err, IsNil)
}

Loading…
Cancel
Save