mirror of
https://github.com/minio/minio.git
synced 2025-01-12 15:33:22 -05:00
Add disk detection for Linux, add new RPC service GetDiskInfoService(), remove dummy HelloService()
This commit is contained in:
parent
181727ab57
commit
e66a84242a
@ -117,9 +117,9 @@ func getAPIHandler(conf api.Config) (http.Handler, api.Minio) {
|
||||
func getRPCHandler() http.Handler {
|
||||
s := rpc.NewServer()
|
||||
s.RegisterJSONCodec()
|
||||
s.RegisterService(new(rpc.HelloService), "")
|
||||
s.RegisterService(new(rpc.VersionService), "")
|
||||
s.RegisterService(new(rpc.GetSysInfoService), "")
|
||||
s.RegisterService(new(rpc.GetDiskInfoService), "")
|
||||
// Add new services here
|
||||
return registerRPC(router.NewRouter(), s)
|
||||
}
|
||||
|
58
pkg/server/rpc/diskinfo.go
Normal file
58
pkg/server/rpc/diskinfo.go
Normal file
@ -0,0 +1,58 @@
|
||||
/*
|
||||
* 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"
|
||||
)
|
||||
|
||||
// GetDiskInfoService disk info service
|
||||
type GetDiskInfoService struct{}
|
||||
|
||||
// GetDiskInfoReply disk info reply for disk info service
|
||||
type GetDiskInfoReply struct {
|
||||
Hostname string `json:"hostname"`
|
||||
Disks []string `json:"disks"`
|
||||
DiskAttributes map[string]scsi.Attributes `json:"disk-attrs"`
|
||||
}
|
||||
|
||||
func setDiskInfoReply(sis *GetDiskInfoReply) error {
|
||||
var err error
|
||||
sis.Hostname, err = os.Hostname()
|
||||
if err != nil {
|
||||
return iodine.New(err, nil)
|
||||
}
|
||||
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 *GetDiskInfoService) Get(r *http.Request, args *Args, reply *GetDiskInfoReply) error {
|
||||
return setDiskInfoReply(reply)
|
||||
}
|
@ -20,61 +20,10 @@ import (
|
||||
"net/http"
|
||||
"os"
|
||||
"runtime"
|
||||
"time"
|
||||
|
||||
"github.com/minio/minio/pkg/iodine"
|
||||
)
|
||||
|
||||
// HelloArgs - hello args
|
||||
type HelloArgs struct {
|
||||
Who string
|
||||
}
|
||||
|
||||
// HelloReply - hello reply
|
||||
type HelloReply struct {
|
||||
Message string
|
||||
}
|
||||
|
||||
// HelloService - hello service
|
||||
type HelloService struct{}
|
||||
|
||||
// Say method
|
||||
func (h *HelloService) Say(r *http.Request, args *HelloArgs, reply *HelloReply) error {
|
||||
reply.Message = "Hello, " + args.Who + "!"
|
||||
return nil
|
||||
}
|
||||
|
||||
// Args basic json RPC params
|
||||
type Args struct {
|
||||
Request string
|
||||
}
|
||||
|
||||
// VersionReply version reply
|
||||
type VersionReply struct {
|
||||
Version string `json:"version"`
|
||||
BuildDate string `json:"build-date"`
|
||||
}
|
||||
|
||||
// VersionService -
|
||||
type VersionService struct{}
|
||||
|
||||
func getVersion() string {
|
||||
return "0.0.1"
|
||||
}
|
||||
func getBuildDate() string {
|
||||
return time.Now().UTC().Format(http.TimeFormat)
|
||||
}
|
||||
|
||||
func setVersionReply(reply *VersionReply) {
|
||||
reply.Version = getVersion()
|
||||
reply.BuildDate = getBuildDate()
|
||||
return
|
||||
}
|
||||
|
||||
// Get method
|
||||
func (v *VersionService) Get(r *http.Request, args *Args, reply *VersionReply) error {
|
||||
setVersionReply(reply)
|
||||
return nil
|
||||
}
|
||||
|
||||
// GetSysInfoService -
|
||||
type GetSysInfoService struct{}
|
||||
|
||||
@ -95,7 +44,12 @@ func setSysInfoReply(sis *GetSysInfoReply) error {
|
||||
sis.SysCPUS = runtime.NumCPU()
|
||||
sis.Routines = runtime.NumGoroutine()
|
||||
sis.GOVersion = runtime.Version()
|
||||
sis.Hostname, _ = os.Hostname()
|
||||
|
||||
var err error
|
||||
sis.Hostname, err = os.Hostname()
|
||||
if err != nil {
|
||||
return iodine.New(err, nil)
|
||||
}
|
||||
|
||||
var memStats runtime.MemStats
|
||||
runtime.ReadMemStats(&memStats)
|
55
pkg/server/rpc/version.go
Normal file
55
pkg/server/rpc/version.go
Normal file
@ -0,0 +1,55 @@
|
||||
/*
|
||||
* 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"
|
||||
"time"
|
||||
)
|
||||
|
||||
// Args basic json RPC params
|
||||
type Args struct {
|
||||
Request string
|
||||
}
|
||||
|
||||
// VersionReply version reply
|
||||
type VersionReply struct {
|
||||
Version string `json:"version"`
|
||||
BuildDate string `json:"build-date"`
|
||||
}
|
||||
|
||||
// VersionService -
|
||||
type VersionService struct{}
|
||||
|
||||
func getVersion() string {
|
||||
return "0.0.1"
|
||||
}
|
||||
func getBuildDate() string {
|
||||
return time.Now().UTC().Format(http.TimeFormat)
|
||||
}
|
||||
|
||||
func setVersionReply(reply *VersionReply) {
|
||||
reply.Version = getVersion()
|
||||
reply.BuildDate = getBuildDate()
|
||||
return
|
||||
}
|
||||
|
||||
// Get method
|
||||
func (v *VersionService) Get(r *http.Request, args *Args, reply *VersionReply) error {
|
||||
setVersionReply(reply)
|
||||
return nil
|
||||
}
|
@ -22,6 +22,7 @@ import (
|
||||
"net/http"
|
||||
"strings"
|
||||
|
||||
"github.com/minio/minio/pkg/iodine"
|
||||
"github.com/minio/minio/pkg/server/api"
|
||||
)
|
||||
|
||||
@ -38,7 +39,7 @@ func startAPI(errCh chan error, conf api.Config, apiHandler http.Handler) {
|
||||
|
||||
host, port, err := net.SplitHostPort(conf.Address)
|
||||
if err != nil {
|
||||
errCh <- err
|
||||
errCh <- iodine.New(err, nil)
|
||||
return
|
||||
}
|
||||
|
||||
@ -49,7 +50,7 @@ func startAPI(errCh chan error, conf api.Config, apiHandler http.Handler) {
|
||||
default:
|
||||
addrs, err := net.InterfaceAddrs()
|
||||
if err != nil {
|
||||
errCh <- err
|
||||
errCh <- iodine.New(err, nil)
|
||||
return
|
||||
}
|
||||
for _, addr := range addrs {
|
||||
@ -109,8 +110,8 @@ func StartServices(conf api.Config) error {
|
||||
|
||||
select {
|
||||
case err := <-apiErrCh:
|
||||
return err
|
||||
return iodine.New(err, nil)
|
||||
case err := <-rpcErrCh:
|
||||
return err
|
||||
return iodine.New(err, nil)
|
||||
}
|
||||
}
|
||||
|
66
pkg/utils/scsi/constants.go
Normal file
66
pkg/utils/scsi/constants.go
Normal file
@ -0,0 +1,66 @@
|
||||
// +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",
|
||||
}
|
48
pkg/utils/scsi/scsi-common.go
Normal file
48
pkg/utils/scsi/scsi-common.go
Normal file
@ -0,0 +1,48 @@
|
||||
// +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
|
||||
|
||||
import (
|
||||
"io/ioutil"
|
||||
"os"
|
||||
"path"
|
||||
"strings"
|
||||
)
|
||||
|
||||
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
|
||||
}
|
101
pkg/utils/scsi/scsi.go
Normal file
101
pkg/utils/scsi/scsi.go
Normal file
@ -0,0 +1,101 @@
|
||||
// +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
|
||||
|
||||
// Attributes Scsi device attributes
|
||||
type Attributes map[string]string
|
||||
|
||||
// Disks is a list of scsis disks and attributes
|
||||
type Disks map[string]Attributes
|
||||
|
||||
// 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 {
|
||||
return Disks{}, iodine.New(err, 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
|
||||
}
|
36
pkg/utils/scsi/scsi_test.go
Normal file
36
pkg/utils/scsi/scsi_test.go
Normal file
@ -0,0 +1,36 @@
|
||||
// +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 (
|
||||
"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)
|
||||
}
|
Loading…
Reference in New Issue
Block a user