Skip to content

Commit

Permalink
Add support for sending NMI
Browse files Browse the repository at this point in the history
Support for sending an NMI has been added to ipmi, redfish,
redfishwrapper, and all providers that use the redfishwrapper.
  • Loading branch information
coffeefreak101 committed Mar 7, 2024
1 parent 7a00485 commit 279499d
Show file tree
Hide file tree
Showing 11 changed files with 299 additions and 25 deletions.
67 changes: 67 additions & 0 deletions bmc/nmi.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,67 @@
package bmc

import (
"context"
"errors"
"fmt"
"time"

"github.com/hashicorp/go-multierror"
)

type NMISender interface {
SendNMI(ctx context.Context) error
}

// sendNMI will return true on if the call was successful, else false
func sendNMI(ctx context.Context, timeout time.Duration, sender NMISender, metadata *Metadata) error {
senderName := getProviderName(sender)
ctx, cancel := context.WithTimeout(ctx, timeout)
defer cancel()

metadata.ProvidersAttempted = append(metadata.ProvidersAttempted, senderName)

err := sender.SendNMI(ctx)
if err != nil {
metadata.FailedProviderDetail[senderName] = err.Error()
return err
}

metadata.SuccessfulProvider = senderName

return nil
}

// SendNMIFromInterface will look for providers that implement NMISender
// and attempt to call SendNMI until a provider is successful,
// or all providers have been exhausted.
func SendNMIFromInterface(
ctx context.Context,
timeout time.Duration,
providers []interface{},
) (metadata Metadata, err error) {
metadata = newMetadata()

for _, provider := range providers {
sender, ok := provider.(NMISender)
if !ok {
err = multierror.Append(err, fmt.Errorf("not an NMISender implementation: %T", provider))
continue
}

sendNMIErr := sendNMI(ctx, timeout, sender, &metadata)
if sendNMIErr != nil {
err = multierror.Append(err, sendNMIErr)
continue
}
return metadata, nil
}

if len(metadata.ProvidersAttempted) == 0 {
err = multierror.Append(err, errors.New("no NMISender implementations found"))
} else {
err = multierror.Append(err, errors.New("failed to send NMI"))
}

return metadata, err
}
124 changes: 124 additions & 0 deletions bmc/nmi_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,124 @@
package bmc

import (
"context"
"testing"
"time"

"github.com/pkg/errors"
"github.com/stretchr/testify/assert"
)

type mockNMISender struct {
err error
}

func (m *mockNMISender) SendNMI(ctx context.Context) error {
select {
case <-ctx.Done():
return ctx.Err()
default:
return m.err
}
}

func (m *mockNMISender) Name() string {
return "mock"
}

func TestSendNMIFromInterface(t *testing.T) {
testCases := []struct {
name string
mockSenders []interface{}
errMsg string
isTimedout bool
expectedMetadata Metadata
}{
{
name: "success",
mockSenders: []interface{}{&mockNMISender{}},
expectedMetadata: Metadata{
SuccessfulProvider: "mock",
ProvidersAttempted: []string{"mock"},
FailedProviderDetail: make(map[string]string),
},
},
{
name: "success with multiple senders",
mockSenders: []interface{}{
nil,
"foo",
&mockNMISender{err: errors.New("err from sender")},
&mockNMISender{},
},
expectedMetadata: Metadata{
SuccessfulProvider: "mock",
ProvidersAttempted: []string{"mock", "mock"},
FailedProviderDetail: map[string]string{"mock": "err from sender"},
},
},
{
name: "not an nmisender",
mockSenders: []interface{}{nil},
errMsg: "not an NMISender",
expectedMetadata: Metadata{
FailedProviderDetail: make(map[string]string),
},
},
{
name: "no nmisenders",
mockSenders: []interface{}{},
errMsg: "no NMISender implementations found",
expectedMetadata: Metadata{
FailedProviderDetail: make(map[string]string),
},
},
{
name: "timed out",
mockSenders: []interface{}{&mockNMISender{}},
isTimedout: true,
errMsg: "context deadline exceeded",
expectedMetadata: Metadata{
ProvidersAttempted: []string{"mock"},
FailedProviderDetail: map[string]string{"mock": "context deadline exceeded"},
},
},
{
name: "error from nmisender",
mockSenders: []interface{}{&mockNMISender{err: errors.New("foobar")}},
errMsg: "foobar",
expectedMetadata: Metadata{
ProvidersAttempted: []string{"mock"},
FailedProviderDetail: map[string]string{"mock": "foobar"},
},
},
{
name: "error when fail to send",
mockSenders: []interface{}{&mockNMISender{err: errors.New("err from sender")}},
errMsg: "failed to send NMI",
expectedMetadata: Metadata{
ProvidersAttempted: []string{"mock"},
FailedProviderDetail: map[string]string{"mock": "err from sender"},
},
},
}

for _, tt := range testCases {
t.Run(tt.name, func(t *testing.T) {
timeout := time.Second * 60
if tt.isTimedout {
timeout = 0
}

metadata, err := SendNMIFromInterface(context.Background(), timeout, tt.mockSenders)

if tt.errMsg == "" {
assert.NoError(t, err)
} else {
assert.ErrorContains(t, err, tt.errMsg)
}

assert.Equal(t, tt.expectedMetadata, metadata)
})
}
}
26 changes: 19 additions & 7 deletions client.go
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,14 @@ import (
"time"

"dario.cat/mergo"
"github.com/bmc-toolbox/common"
"github.com/go-logr/logr"
"github.com/jacobweinstock/registrar"
"go.opentelemetry.io/otel/attribute"
"go.opentelemetry.io/otel/trace"
oteltrace "go.opentelemetry.io/otel/trace"
tracenoop "go.opentelemetry.io/otel/trace/noop"

"github.com/bmc-toolbox/bmclib/v2/bmc"
"github.com/bmc-toolbox/bmclib/v2/constants"
"github.com/bmc-toolbox/bmclib/v2/internal/httpclient"
Expand All @@ -24,13 +32,6 @@ import (
"github.com/bmc-toolbox/bmclib/v2/providers/redfish"
"github.com/bmc-toolbox/bmclib/v2/providers/rpc"
"github.com/bmc-toolbox/bmclib/v2/providers/supermicro"
"github.com/bmc-toolbox/common"
"github.com/go-logr/logr"
"github.com/jacobweinstock/registrar"
"go.opentelemetry.io/otel/attribute"
"go.opentelemetry.io/otel/trace"
oteltrace "go.opentelemetry.io/otel/trace"
tracenoop "go.opentelemetry.io/otel/trace/noop"
)

const (
Expand Down Expand Up @@ -717,3 +718,14 @@ func (c *Client) GetSystemEventLogRaw(ctx context.Context) (eventlog string, err
c.setMetadata(metadata)
return eventlog, err
}

// SendNMI tells the BMC to issue an NMI to the device
func (c *Client) SendNMI(ctx context.Context) error {
ctx, span := c.traceprovider.Tracer(pkgName).Start(ctx, "SendNMI")
defer span.End()

Check warning on line 725 in client.go

View check run for this annotation

Codecov / codecov/patch

client.go#L723-L725

Added lines #L723 - L725 were not covered by tests

metadata, err := bmc.SendNMIFromInterface(ctx, c.perProviderTimeout(ctx), c.registry().GetDriverInterfaces())
c.setMetadata(metadata)

Check warning on line 728 in client.go

View check run for this annotation

Codecov / codecov/patch

client.go#L727-L728

Added lines #L727 - L728 were not covered by tests

return err

Check warning on line 730 in client.go

View check run for this annotation

Codecov / codecov/patch

client.go#L730

Added line #L730 was not covered by tests
}
10 changes: 10 additions & 0 deletions internal/ipmi/ipmi.go
Original file line number Diff line number Diff line change
Expand Up @@ -437,3 +437,13 @@ func (i *Ipmi) DeactivateSOL(ctx context.Context) (err error) {
}
return err
}

// SendPowerDiag tells the BMC to issue an NMI to the device
func (i *Ipmi) SendPowerDiag(ctx context.Context) error {
_, err := i.run(ctx, []string{"chassis", "power", "diag"})
if err != nil {
err = errors.Wrap(err, "failed sending power diag")
}

return err
}
20 changes: 18 additions & 2 deletions internal/redfishwrapper/power.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,9 +6,10 @@ import (
"strings"
"time"

bmclibErrs "github.com/bmc-toolbox/bmclib/v2/errors"
"github.com/pkg/errors"
rf "github.com/stmcginnis/gofish/redfish"

bmclibErrs "github.com/bmc-toolbox/bmclib/v2/errors"
)

// PowerSet sets the power state of a server
Expand Down Expand Up @@ -212,7 +213,6 @@ func (c *Client) SystemForceOff(ctx context.Context) (ok bool, err error) {

system.DisableEtagMatch(c.disableEtagMatch)


err = system.Reset(rf.ForceOffResetType)
if err != nil {
return false, err
Expand All @@ -221,3 +221,19 @@ func (c *Client) SystemForceOff(ctx context.Context) (ok bool, err error) {

return true, nil
}

// SendNMI tells the BMC to issue an NMI to the device
func (c *Client) SendNMI(_ context.Context) error {
ss, err := c.client.Service.Systems()
if err != nil {
return err

Check warning on line 229 in internal/redfishwrapper/power.go

View check run for this annotation

Codecov / codecov/patch

internal/redfishwrapper/power.go#L226-L229

Added lines #L226 - L229 were not covered by tests
}

for _, system := range ss {
if err = system.Reset(rf.NmiResetType); err != nil {
return err

Check warning on line 234 in internal/redfishwrapper/power.go

View check run for this annotation

Codecov / codecov/patch

internal/redfishwrapper/power.go#L232-L234

Added lines #L232 - L234 were not covered by tests
}
}

return nil

Check warning on line 238 in internal/redfishwrapper/power.go

View check run for this annotation

Codecov / codecov/patch

internal/redfishwrapper/power.go#L238

Added line #L238 was not covered by tests
}
12 changes: 9 additions & 3 deletions providers/dell/idrac.go
Original file line number Diff line number Diff line change
Expand Up @@ -10,14 +10,15 @@ import (
"net/http"
"strings"

"github.com/bmc-toolbox/bmclib/v2/internal/httpclient"
"github.com/bmc-toolbox/bmclib/v2/internal/redfishwrapper"
"github.com/bmc-toolbox/bmclib/v2/providers"
"github.com/bmc-toolbox/common"
"github.com/go-logr/logr"
"github.com/jacobweinstock/registrar"
"github.com/pkg/errors"

"github.com/bmc-toolbox/bmclib/v2/internal/httpclient"
"github.com/bmc-toolbox/bmclib/v2/internal/redfishwrapper"
"github.com/bmc-toolbox/bmclib/v2/providers"

bmclibErrs "github.com/bmc-toolbox/bmclib/v2/errors"
)

Expand Down Expand Up @@ -219,6 +220,11 @@ func (c *Conn) BmcReset(ctx context.Context, resetType string) (ok bool, err err
return c.redfishwrapper.BMCReset(ctx, resetType)
}

// SendNMI tells the BMC to issue an NMI to the device
func (c *Conn) SendNMI(ctx context.Context) error {
return c.redfishwrapper.SendNMI(ctx)

Check warning on line 225 in providers/dell/idrac.go

View check run for this annotation

Codecov / codecov/patch

providers/dell/idrac.go#L224-L225

Added lines #L224 - L225 were not covered by tests
}

// deviceManufacturer returns the device manufacturer and model attributes
func (c *Conn) deviceManufacturer(ctx context.Context) (vendor string, err error) {
systems, err := c.redfishwrapper.Systems()
Expand Down
10 changes: 8 additions & 2 deletions providers/ipmitool/ipmitool.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,11 +5,12 @@ import (
"errors"
"strings"

"github.com/go-logr/logr"
"github.com/jacobweinstock/registrar"

bmclibErrs "github.com/bmc-toolbox/bmclib/v2/errors"
"github.com/bmc-toolbox/bmclib/v2/internal/ipmi"
"github.com/bmc-toolbox/bmclib/v2/providers"
"github.com/go-logr/logr"
"github.com/jacobweinstock/registrar"
)

const (
Expand Down Expand Up @@ -201,3 +202,8 @@ func (c *Conn) GetSystemEventLog(ctx context.Context) (entries [][]string, err e
func (c *Conn) GetSystemEventLogRaw(ctx context.Context) (eventlog string, err error) {
return c.ipmitool.GetSystemEventLogRaw(ctx)
}

// SendNMI tells the BMC to issue an NMI to the device
func (c *Conn) SendNMI(ctx context.Context) error {
return c.ipmitool.SendPowerDiag(ctx)

Check warning on line 208 in providers/ipmitool/ipmitool.go

View check run for this annotation

Codecov / codecov/patch

providers/ipmitool/ipmitool.go#L207-L208

Added lines #L207 - L208 were not covered by tests
}
18 changes: 18 additions & 0 deletions providers/ipmitool/ipmitool_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -178,3 +178,21 @@ func TestSystemEventLogGetRaw(t *testing.T) {
t.Log(eventlog)
t.Fatal()
}

func TestSendNMI(t *testing.T) {
t.Skip("need real ipmi server")
host := "127.0.0.1"
port := "623"
user := "ADMIN"
pass := "ADMIN"
i, err := New(host, user, pass, WithPort(port), WithLogger(logging.DefaultLogger()))
if err != nil {
t.Fatal(err)
}
err = i.SendNMI(context.Background())
if err != nil {
t.Fatal(err)
}
t.Log("NMI sent")
t.Fatal()
}
12 changes: 9 additions & 3 deletions providers/openbmc/openbmc.go
Original file line number Diff line number Diff line change
Expand Up @@ -8,13 +8,14 @@ import (
"net/http"
"strings"

"github.com/bmc-toolbox/bmclib/v2/internal/httpclient"
"github.com/bmc-toolbox/bmclib/v2/internal/redfishwrapper"
"github.com/bmc-toolbox/bmclib/v2/providers"
"github.com/bmc-toolbox/common"
"github.com/go-logr/logr"
"github.com/jacobweinstock/registrar"
"github.com/pkg/errors"

"github.com/bmc-toolbox/bmclib/v2/internal/httpclient"
"github.com/bmc-toolbox/bmclib/v2/internal/redfishwrapper"
"github.com/bmc-toolbox/bmclib/v2/providers"
)

const (
Expand Down Expand Up @@ -184,3 +185,8 @@ func (c *Conn) Inventory(ctx context.Context) (device *common.Device, err error)
func (c *Conn) BmcReset(ctx context.Context, resetType string) (ok bool, err error) {
return c.redfishwrapper.BMCReset(ctx, resetType)
}

// SendNMI tells the BMC to issue an NMI to the device
func (c *Conn) SendNMI(ctx context.Context) error {
return c.redfishwrapper.SendNMI(ctx)
}
Loading

0 comments on commit 279499d

Please sign in to comment.