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 {
|
func getRPCHandler() http.Handler {
|
||||||
s := rpc.NewServer()
|
s := rpc.NewServer()
|
||||||
s.RegisterJSONCodec()
|
s.RegisterJSONCodec()
|
||||||
s.RegisterService(new(rpc.HelloService), "")
|
|
||||||
s.RegisterService(new(rpc.VersionService), "")
|
s.RegisterService(new(rpc.VersionService), "")
|
||||||
s.RegisterService(new(rpc.GetSysInfoService), "")
|
s.RegisterService(new(rpc.GetSysInfoService), "")
|
||||||
|
s.RegisterService(new(rpc.GetDiskInfoService), "")
|
||||||
// Add new services here
|
// Add new services here
|
||||||
return registerRPC(router.NewRouter(), s)
|
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"
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
"runtime"
|
"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 -
|
// GetSysInfoService -
|
||||||
type GetSysInfoService struct{}
|
type GetSysInfoService struct{}
|
||||||
|
|
||||||
@ -95,7 +44,12 @@ func setSysInfoReply(sis *GetSysInfoReply) error {
|
|||||||
sis.SysCPUS = runtime.NumCPU()
|
sis.SysCPUS = runtime.NumCPU()
|
||||||
sis.Routines = runtime.NumGoroutine()
|
sis.Routines = runtime.NumGoroutine()
|
||||||
sis.GOVersion = runtime.Version()
|
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
|
var memStats runtime.MemStats
|
||||||
runtime.ReadMemStats(&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"
|
"net/http"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/minio/minio/pkg/iodine"
|
||||||
"github.com/minio/minio/pkg/server/api"
|
"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)
|
host, port, err := net.SplitHostPort(conf.Address)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
errCh <- err
|
errCh <- iodine.New(err, nil)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -49,7 +50,7 @@ func startAPI(errCh chan error, conf api.Config, apiHandler http.Handler) {
|
|||||||
default:
|
default:
|
||||||
addrs, err := net.InterfaceAddrs()
|
addrs, err := net.InterfaceAddrs()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
errCh <- err
|
errCh <- iodine.New(err, nil)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
for _, addr := range addrs {
|
for _, addr := range addrs {
|
||||||
@ -109,8 +110,8 @@ func StartServices(conf api.Config) error {
|
|||||||
|
|
||||||
select {
|
select {
|
||||||
case err := <-apiErrCh:
|
case err := <-apiErrCh:
|
||||||
return err
|
return iodine.New(err, nil)
|
||||||
case err := <-rpcErrCh:
|
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