1
0
mirror of https://github.com/hybridgroup/gobot.git synced 2025-04-29 13:49:14 +08:00
hybridgroup.gobot/platforms/chip/chip_adaptor_test.go

273 lines
7.3 KiB
Go
Raw Normal View History

package chip
import (
"fmt"
"strconv"
"strings"
"testing"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
"gobot.io/x/gobot/v2"
"gobot.io/x/gobot/v2/drivers/gpio"
"gobot.io/x/gobot/v2/drivers/i2c"
"gobot.io/x/gobot/v2/platforms/adaptors"
"gobot.io/x/gobot/v2/system"
)
2022-11-20 19:22:26 +01:00
// make sure that this Adaptor fulfills all the required interfaces
var (
_ gobot.Adaptor = (*Adaptor)(nil)
_ gobot.DigitalPinnerProvider = (*Adaptor)(nil)
_ gobot.PWMPinnerProvider = (*Adaptor)(nil)
_ gpio.DigitalReader = (*Adaptor)(nil)
_ gpio.DigitalWriter = (*Adaptor)(nil)
_ gpio.PwmWriter = (*Adaptor)(nil)
_ gpio.ServoWriter = (*Adaptor)(nil)
_ i2c.Connector = (*Adaptor)(nil)
)
2022-11-05 07:42:28 +01:00
var mockPaths = []string{
"/sys/class/gpio/export",
"/sys/class/gpio/unexport",
"/sys/class/gpio/gpio50/value",
"/sys/class/gpio/gpio50/direction",
"/sys/class/gpio/gpio139/value",
"/sys/class/gpio/gpio139/direction",
"/sys/class/pwm/pwmchip0/export",
"/sys/class/pwm/pwmchip0/unexport",
"/sys/class/pwm/pwmchip0/pwm0/enable",
"/sys/class/pwm/pwmchip0/pwm0/duty_cycle",
"/sys/class/pwm/pwmchip0/pwm0/polarity",
"/sys/class/pwm/pwmchip0/pwm0/period",
}
2022-11-20 19:22:26 +01:00
func initTestAdaptorWithMockedFilesystem() (*Adaptor, *system.MockFilesystem) {
a := NewAdaptor()
2022-11-20 19:22:26 +01:00
fs := a.sys.UseMockFilesystem(mockPaths)
if err := a.Connect(); err != nil {
panic(err)
}
return a, fs
}
2022-11-20 19:22:26 +01:00
func initTestProAdaptorWithMockedFilesystem() (*Adaptor, *system.MockFilesystem) {
a := NewProAdaptor()
2022-11-20 19:22:26 +01:00
fs := a.sys.UseMockFilesystem(mockPaths)
if err := a.Connect(); err != nil {
panic(err)
}
return a, fs
}
2022-11-05 07:42:28 +01:00
func TestName(t *testing.T) {
a := NewAdaptor()
assert.True(t, strings.HasPrefix(a.Name(), "CHIP"))
a.SetName("NewName")
assert.Equal(t, "NewName", a.Name())
}
func TestNewProAdaptor(t *testing.T) {
a := NewProAdaptor()
assert.True(t, strings.HasPrefix(a.Name(), "CHIP Pro"))
}
2022-11-05 07:42:28 +01:00
func TestFinalizeErrorAfterGPIO(t *testing.T) {
a, fs := initTestAdaptorWithMockedFilesystem()
require.NoError(t, a.DigitalWrite("CSID7", 1))
fs.WithWriteError = true
err := a.Finalize()
require.ErrorContains(t, err, "write error")
}
2022-11-05 07:42:28 +01:00
func TestFinalizeErrorAfterPWM(t *testing.T) {
a, fs := initTestAdaptorWithMockedFilesystem()
fs.Files["/sys/class/pwm/pwmchip0/pwm0/duty_cycle"].Contents = "0"
fs.Files["/sys/class/pwm/pwmchip0/pwm0/period"].Contents = "0"
require.NoError(t, a.PwmWrite("PWM0", 100))
fs.WithWriteError = true
err := a.Finalize()
require.ErrorContains(t, err, "write error")
}
2022-11-05 07:42:28 +01:00
func TestDigitalIO(t *testing.T) {
a, fs := initTestAdaptorWithMockedFilesystem()
require.NoError(t, a.DigitalWrite("CSID7", 1))
assert.Equal(t, "1", fs.Files["/sys/class/gpio/gpio139/value"].Contents)
fs.Files["/sys/class/gpio/gpio50/value"].Contents = "1"
i, _ := a.DigitalRead("TWI2-SDA")
assert.Equal(t, 1, i)
require.ErrorContains(t, a.DigitalWrite("XIO-P10", 1), "'XIO-P10' is not a valid id for a digital pin")
require.NoError(t, a.Finalize())
}
2022-11-05 07:42:28 +01:00
func TestProDigitalIO(t *testing.T) {
a, fs := initTestProAdaptorWithMockedFilesystem()
_ = a.Connect()
_ = a.DigitalWrite("CSID7", 1)
assert.Equal(t, "1", fs.Files["/sys/class/gpio/gpio139/value"].Contents)
fs.Files["/sys/class/gpio/gpio50/value"].Contents = "1"
i, _ := a.DigitalRead("TWI2-SDA")
assert.Equal(t, 1, i)
require.ErrorContains(t, a.DigitalWrite("XIO-P0", 1), "'XIO-P0' is not a valid id for a digital pin")
require.NoError(t, a.Finalize())
}
func TestPWMWrite(t *testing.T) {
// arrange
2022-11-05 07:42:28 +01:00
a, fs := initTestAdaptorWithMockedFilesystem()
fs.Files["/sys/class/pwm/pwmchip0/pwm0/duty_cycle"].Contents = "0"
fs.Files["/sys/class/pwm/pwmchip0/pwm0/period"].Contents = "0"
// act
err := a.PwmWrite("PWM0", 100)
// assert
require.NoError(t, err)
assert.Equal(t, "0", fs.Files["/sys/class/pwm/pwmchip0/export"].Contents)
assert.Equal(t, "1", fs.Files["/sys/class/pwm/pwmchip0/pwm0/enable"].Contents)
assert.Equal(t, "3921568", fs.Files["/sys/class/pwm/pwmchip0/pwm0/duty_cycle"].Contents)
assert.Equal(t, "10000000", fs.Files["/sys/class/pwm/pwmchip0/pwm0/period"].Contents) // pwmPeriodDefault
assert.Equal(t, "normal", fs.Files["/sys/class/pwm/pwmchip0/pwm0/polarity"].Contents)
require.NoError(t, a.Finalize())
}
func TestServoWrite(t *testing.T) {
// arrange: prepare 50Hz for servos
const (
pin = "PWM0"
fiftyHzNano = 20000000
)
a := NewAdaptor(adaptors.WithPWMDefaultPeriodForPin(pin, fiftyHzNano))
fs := a.sys.UseMockFilesystem(mockPaths)
require.NoError(t, a.Connect())
fs.Files["/sys/class/pwm/pwmchip0/pwm0/duty_cycle"].Contents = "0"
fs.Files["/sys/class/pwm/pwmchip0/pwm0/period"].Contents = "0"
// act & assert for 0° (min default value)
err := a.ServoWrite(pin, 0)
require.NoError(t, err)
assert.Equal(t, strconv.Itoa(fiftyHzNano), fs.Files["/sys/class/pwm/pwmchip0/pwm0/period"].Contents)
assert.Equal(t, "500000", fs.Files["/sys/class/pwm/pwmchip0/pwm0/duty_cycle"].Contents)
// act & assert for 180° (max default value)
err = a.ServoWrite(pin, 180)
require.NoError(t, err)
assert.Equal(t, strconv.Itoa(fiftyHzNano), fs.Files["/sys/class/pwm/pwmchip0/pwm0/period"].Contents)
assert.Equal(t, "2500000", fs.Files["/sys/class/pwm/pwmchip0/pwm0/duty_cycle"].Contents)
require.NoError(t, a.Finalize())
}
func TestI2cDefaultBus(t *testing.T) {
a := NewAdaptor()
assert.Equal(t, 1, a.DefaultI2cBus())
}
func TestI2cFinalizeWithErrors(t *testing.T) {
// arrange
a := NewAdaptor()
a.sys.UseMockSyscall()
fs := a.sys.UseMockFilesystem([]string{"/dev/i2c-2"})
require.NoError(t, a.Connect())
con, err := a.GetI2cConnection(0xff, 2)
require.NoError(t, err)
_, err = con.Write([]byte{0xbf})
require.NoError(t, err)
fs.WithCloseError = true
// act
err = a.Finalize()
// assert
require.ErrorContains(t, err, "close error")
}
func Test_validateI2cBusNumber(t *testing.T) {
tests := map[string]struct {
busNr int
wantErr error
}{
"number_negative_error": {
busNr: -1,
wantErr: fmt.Errorf("Bus number -1 out of range"),
},
"number_0_ok": {
busNr: 0,
},
"number_1_ok": {
busNr: 1,
},
"number_2_ok": {
busNr: 2,
},
"number_3_error": {
busNr: 3,
wantErr: fmt.Errorf("Bus number 3 out of range"),
},
}
for name, tc := range tests {
t.Run(name, func(t *testing.T) {
// arrange
a := NewAdaptor()
// act
err := a.validateI2cBusNumber(tc.busNr)
// assert
assert.Equal(t, tc.wantErr, err)
})
}
}
func Test_translatePWMPin(t *testing.T) {
tests := map[string]struct {
usePro bool
wantDir string
wantChannel int
wantErr error
}{
"PWM0": {
wantDir: "/sys/class/pwm/pwmchip0",
wantChannel: 0,
},
"PWM1": {
usePro: true,
wantDir: "/sys/class/pwm/pwmchip0",
wantChannel: 1,
},
"33_1": {
wantDir: "",
wantChannel: -1,
wantErr: fmt.Errorf("'33_1' is not a valid id for a pin"),
},
"AP-EINT3": {
wantDir: "",
wantChannel: -1,
wantErr: fmt.Errorf("'AP-EINT3' is not a valid id for a PWM pin"),
},
}
for name, tc := range tests {
t.Run(name, func(t *testing.T) {
// arrange
var a *Adaptor
if tc.usePro {
a = NewProAdaptor()
} else {
a = NewAdaptor()
}
// act
dir, channel, err := a.translatePWMPin(name)
// assert
assert.Equal(t, tc.wantErr, err)
assert.Equal(t, tc.wantDir, dir)
assert.Equal(t, tc.wantChannel, channel)
})
}
}