master
parent
49c705da99
commit
4498662c16
@ -0,0 +1 @@ |
||||
donut |
@ -1 +0,0 @@ |
||||
mkdonut |
@ -1,67 +0,0 @@ |
||||
/* |
||||
* Minimalist Object Storage, (C) 2015 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 rpc |
||||
|
||||
import ( |
||||
"net/http" |
||||
"os" |
||||
|
||||
"github.com/minio/minio/pkg/iodine" |
||||
"github.com/minio/minio/pkg/utils/scsi" |
||||
) |
||||
|
||||
// DiskInfoService disk info service
|
||||
type DiskInfoService struct{} |
||||
|
||||
// DiskInfoReply disk info reply for disk info service
|
||||
type DiskInfoReply struct { |
||||
Hostname string `json:"hostname"` |
||||
Mounts map[string]scsi.Mountinfo `json:"mounts"` |
||||
Disks []string `json:"disks"` |
||||
DiskAttributes map[string]scsi.Attributes `json:"-"` // for the time being not unmarshalling this
|
||||
} |
||||
|
||||
func setDiskInfoReply(sis *DiskInfoReply) error { |
||||
var err error |
||||
sis.Hostname, err = os.Hostname() |
||||
if 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() |
||||
if err != nil { |
||||
return iodine.New(err, nil) |
||||
} |
||||
sis.DiskAttributes = make(map[string]scsi.Attributes) |
||||
|
||||
for k, v := range disks { |
||||
sis.Disks = append(sis.Disks, k) |
||||
sis.DiskAttributes[k] = v |
||||
} |
||||
return nil |
||||
} |
||||
|
||||
// Get method
|
||||
func (s *DiskInfoService) Get(r *http.Request, args *Args, reply *DiskInfoReply) error { |
||||
return setDiskInfoReply(reply) |
||||
} |
@ -1,66 +0,0 @@ |
||||
// +build linux,amd64
|
||||
|
||||
/* |
||||
* 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 |
||||
|
||||
// System sysfs constants
|
||||
const ( |
||||
// From 2.6.x kernel onwards, no need to support procfs
|
||||
sysFSROOT = "/sys" |
||||
sysFSDEVICES = "/sys/bus/scsi/devices" |
||||
sysFSBLOCK = "/block" |
||||
sysFSCLASSSCSIDEVICES = "/sys/class/scsi_device" |
||||
udev = "/dev" |
||||
devDISKBYID = "/dev/disk/by-id" |
||||
) |
||||
|
||||
// ScsiDEVICETYPES list of various scsi devices
|
||||
var ScsiDEVICETYPES = []string{ |
||||
"disk", |
||||
"tape", |
||||
"printer", |
||||
"process", |
||||
"worm", |
||||
"cd/dvd", |
||||
"scanner", |
||||
"optical", |
||||
"mediumx", |
||||
"comms", |
||||
"(0xa)", |
||||
"(0xb)", |
||||
"storage", |
||||
"enclosu", |
||||
"sim disk", |
||||
"optical rd", |
||||
"bridge", |
||||
"osd", |
||||
"adi", |
||||
"sec man", |
||||
"zbc", |
||||
"(0x15)", |
||||
"(0x16)", |
||||
"(0x17)", |
||||
"(0x18)", |
||||
"(0x19)", |
||||
"(0x1a)", |
||||
"(0x1b)", |
||||
"(0x1c)", |
||||
"(0x1e)", |
||||
"wlun", |
||||
"no dev", |
||||
} |
@ -1,112 +0,0 @@ |
||||
// +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" |
||||
) |
||||
|
||||
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 |
||||
} |
@ -1,23 +0,0 @@ |
||||
/* |
||||
* 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 |
||||
|
||||
// GetMountInfo - get mount info map
|
||||
func GetMountInfo() (map[string]Mountinfo, error) { |
||||
// Stub implementation; returns an empty map
|
||||
return make(map[string]Mountinfo), nil |
||||
} |
@ -1,62 +0,0 @@ |
||||
/* |
||||
* 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 ( |
||||
"io/ioutil" |
||||
"os" |
||||
"path" |
||||
"strings" |
||||
) |
||||
|
||||
// Attributes Scsi device attributes
|
||||
type Attributes map[string]string |
||||
|
||||
// Disks is a list of scsis disks and attributes
|
||||
type Disks map[string]Attributes |
||||
|
||||
// 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 */ |
||||
} |
||||
|
||||
func getattrs(scsiAttrPath string, scsiAttrList []string) map[string]string { |
||||
attrMap := make(map[string]string) |
||||
for _, attr := range scsiAttrList { |
||||
value, _ := ioutil.ReadFile(path.Join(scsiAttrPath, attr)) |
||||
attrMap[attr] = strings.Trim(string(value[:]), "\n") // remove odd newlines
|
||||
} |
||||
return attrMap |
||||
} |
||||
|
||||
func filterdisks(files []os.FileInfo) (scsidisks []string) { |
||||
for _, fi := range files { |
||||
if strings.Contains(fi.Name(), "host") { |
||||
continue |
||||
} |
||||
if strings.Contains(fi.Name(), "target") { |
||||
continue |
||||
} |
||||
scsidisks = append(scsidisks, fi.Name()) |
||||
} |
||||
return scsidisks |
||||
} |
@ -1,96 +0,0 @@ |
||||
// +build linux,amd64
|
||||
|
||||
/* |
||||
* Minimalist Object Storage, (C) 2015 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 ( |
||||
"errors" |
||||
"io/ioutil" |
||||
"path/filepath" |
||||
|
||||
"github.com/minio/minio/pkg/iodine" |
||||
) |
||||
|
||||
// NOTE : supporting virtio based scsi devices is out of scope for this implementation
|
||||
|
||||
// Get get disk scsi params
|
||||
func (d Disks) Get(disk string) Attributes { |
||||
return d[disk] |
||||
} |
||||
|
||||
// Len return len of total disks
|
||||
func (d Disks) Len() int { |
||||
return len(d) |
||||
} |
||||
|
||||
// GetDisks - get system devices list
|
||||
func GetDisks() (Disks, error) { |
||||
var scsidevices []string |
||||
var scsiAttrList []string |
||||
d := Disks{} |
||||
|
||||
sysFiles, err := ioutil.ReadDir(sysFSDEVICES) |
||||
if err != nil { |
||||
// may be an amazon instance, ignore this
|
||||
return Disks{}, nil |
||||
} |
||||
|
||||
scsidevices = filterdisks(sysFiles) |
||||
if len(scsidevices) == 0 { |
||||
// may be a docker instance, ignore this
|
||||
return Disks{}, nil |
||||
} |
||||
|
||||
for _, s := range scsidevices { |
||||
scsiAttrPath := filepath.Join(sysFSDEVICES, s) |
||||
scsiAttrs, err := ioutil.ReadDir(scsiAttrPath) |
||||
if err != nil { |
||||
return Disks{}, iodine.New(err, nil) |
||||
} |
||||
scsiBlockPath := filepath.Join(sysFSDEVICES, s, sysFSBLOCK) |
||||
scsidevList, err := ioutil.ReadDir(scsiBlockPath) |
||||
if err != nil { |
||||
return Disks{}, iodine.New(err, nil) |
||||
} |
||||
if len(scsidevList) > 1 { |
||||
return Disks{}, iodine.New(errors.New("Scsi address points to multiple block devices"), nil) |
||||
} |
||||
device := filepath.Join(udev, scsidevList[0].Name()) |
||||
for _, sa := range scsiAttrs { |
||||
// Skip directories
|
||||
if sa.IsDir() { |
||||
continue |
||||
} |
||||
// Skip symlinks
|
||||
if !sa.Mode().IsRegular() { |
||||
continue |
||||
} |
||||
// Skip, not readable, write-only
|
||||
if sa.Mode().Perm() == 128 { |
||||
continue |
||||
} |
||||
scsiAttrList = append(scsiAttrList, sa.Name()) |
||||
} |
||||
|
||||
if len(scsiAttrList) == 0 { |
||||
return Disks{}, iodine.New(errors.New("No scsi attributes found"), nil) |
||||
} |
||||
d[device] = getattrs(scsiAttrPath, scsiAttrList) |
||||
} |
||||
return d, nil |
||||
} |
@ -1,23 +0,0 @@ |
||||
/* |
||||
* Minimalist Object Storage, (C) 2015 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 |
||||
|
||||
// GetDisks - get system devices list
|
||||
func GetDisks() (Disks, error) { |
||||
// Stub implementation; returns empty disk information
|
||||
return Disks{}, nil |
||||
} |
@ -1,39 +0,0 @@ |
||||
/* |
||||
* Minimalist Object Storage, (C) 2015 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 ( |
||||
"testing" |
||||
|
||||
. "github.com/minio/check" |
||||
) |
||||
|
||||
type MySuite struct{} |
||||
|
||||
var _ = Suite(&MySuite{}) |
||||
|
||||
func Test(t *testing.T) { TestingT(t) } |
||||
|
||||
func (s *MySuite) TestSCSI(c *C) { |
||||
_, err := GetDisks() |
||||
c.Assert(err, IsNil) |
||||
} |
||||
|
||||
func (s *MySuite) TestMountInfo(c *C) { |
||||
_, err := GetMountInfo() |
||||
c.Assert(err, IsNil) |
||||
} |
Loading…
Reference in new issue