mirror of
https://github.com/httprunner/httprunner.git
synced 2026-05-11 10:00:23 +08:00
659 lines
17 KiB
Go
659 lines
17 KiB
Go
package hrp
|
|
|
|
import (
|
|
"fmt"
|
|
|
|
"github.com/pkg/errors"
|
|
"github.com/rs/zerolog/log"
|
|
|
|
"github.com/httprunner/httprunner/v4/hrp/internal/builtin"
|
|
"github.com/httprunner/httprunner/v4/hrp/internal/code"
|
|
"github.com/httprunner/httprunner/v4/hrp/pkg/uixt"
|
|
)
|
|
|
|
type MobileStep struct {
|
|
Serial string `json:"serial,omitempty" yaml:"serial,omitempty"`
|
|
uixt.MobileAction `yaml:",inline"`
|
|
Actions []uixt.MobileAction `json:"actions,omitempty" yaml:"actions,omitempty"`
|
|
}
|
|
|
|
// StepMobile implements IStep interface.
|
|
type StepMobile struct {
|
|
step *TStep
|
|
}
|
|
|
|
func (s *StepMobile) mobileStep() *MobileStep {
|
|
if s.step.IOS != nil {
|
|
return s.step.IOS
|
|
}
|
|
return s.step.Android
|
|
}
|
|
|
|
func (s *StepMobile) Serial(serial string) *StepMobile {
|
|
s.mobileStep().Serial = serial
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
func (s *StepMobile) InstallApp(path string) *StepMobile {
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, uixt.MobileAction{
|
|
Method: uixt.AppInstall,
|
|
Params: path,
|
|
})
|
|
return s
|
|
}
|
|
|
|
func (s *StepMobile) AppLaunch(bundleId string) *StepMobile {
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, uixt.MobileAction{
|
|
Method: uixt.AppLaunch,
|
|
Params: bundleId,
|
|
})
|
|
return s
|
|
}
|
|
|
|
func (s *StepMobile) AppTerminate(bundleId string) *StepMobile {
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, uixt.MobileAction{
|
|
Method: uixt.AppTerminate,
|
|
Params: bundleId,
|
|
})
|
|
return s
|
|
}
|
|
|
|
func (s *StepMobile) Home() *StepMobile {
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, uixt.MobileAction{
|
|
Method: uixt.ACTION_Home,
|
|
Params: nil,
|
|
})
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
// TapXY taps the point {X,Y}, X & Y is percentage of coordinates
|
|
func (s *StepMobile) TapXY(x, y float64, options ...uixt.ActionOption) *StepMobile {
|
|
action := uixt.MobileAction{
|
|
Method: uixt.ACTION_TapXY,
|
|
Params: []float64{x, y},
|
|
}
|
|
for _, option := range options {
|
|
option(&action)
|
|
}
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, action)
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
// TapAbsXY taps the point {X,Y}, X & Y is absolute coordinates
|
|
func (s *StepMobile) TapAbsXY(x, y float64, options ...uixt.ActionOption) *StepMobile {
|
|
action := uixt.MobileAction{
|
|
Method: uixt.ACTION_TapAbsXY,
|
|
Params: []float64{x, y},
|
|
}
|
|
for _, option := range options {
|
|
option(&action)
|
|
}
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, action)
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
// Tap taps on the target element
|
|
func (s *StepMobile) Tap(params string, options ...uixt.ActionOption) *StepMobile {
|
|
action := uixt.MobileAction{
|
|
Method: uixt.ACTION_Tap,
|
|
Params: params,
|
|
}
|
|
for _, option := range options {
|
|
option(&action)
|
|
}
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, action)
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
// Tap taps on the target element by OCR recognition
|
|
func (s *StepMobile) TapByOCR(ocrText string, options ...uixt.ActionOption) *StepMobile {
|
|
action := uixt.MobileAction{
|
|
Method: uixt.ACTION_TapByOCR,
|
|
Params: ocrText,
|
|
}
|
|
for _, option := range options {
|
|
option(&action)
|
|
}
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, action)
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
// Tap taps on the target element by CV recognition
|
|
func (s *StepMobile) TapByCV(imagePath string, options ...uixt.ActionOption) *StepMobile {
|
|
action := uixt.MobileAction{
|
|
Method: uixt.ACTION_TapByCV,
|
|
Params: imagePath,
|
|
}
|
|
for _, option := range options {
|
|
option(&action)
|
|
}
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, action)
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
// DoubleTapXY double taps the point {X,Y}, X & Y is percentage of coordinates
|
|
func (s *StepMobile) DoubleTapXY(x, y float64) *StepMobile {
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, uixt.MobileAction{
|
|
Method: uixt.ACTION_DoubleTapXY,
|
|
Params: []float64{x, y},
|
|
})
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
func (s *StepMobile) DoubleTap(params string, options ...uixt.ActionOption) *StepMobile {
|
|
action := uixt.MobileAction{
|
|
Method: uixt.ACTION_DoubleTap,
|
|
Params: params,
|
|
}
|
|
for _, option := range options {
|
|
option(&action)
|
|
}
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, action)
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
func (s *StepMobile) Back(options ...uixt.ActionOption) *StepMobile {
|
|
action := uixt.MobileAction{
|
|
Method: uixt.ACTION_Back,
|
|
Params: nil,
|
|
}
|
|
for _, option := range options {
|
|
option(&action)
|
|
}
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, action)
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
func (s *StepMobile) Swipe(sx, sy, ex, ey float64, options ...uixt.ActionOption) *StepMobile {
|
|
action := uixt.MobileAction{
|
|
Method: uixt.ACTION_Swipe,
|
|
Params: []float64{sx, sy, ex, ey},
|
|
}
|
|
for _, option := range options {
|
|
option(&action)
|
|
}
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, action)
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
func (s *StepMobile) SwipeUp(options ...uixt.ActionOption) *StepMobile {
|
|
action := uixt.MobileAction{
|
|
Method: uixt.ACTION_Swipe,
|
|
Params: "up",
|
|
}
|
|
for _, option := range options {
|
|
option(&action)
|
|
}
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, action)
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
func (s *StepMobile) SwipeDown(options ...uixt.ActionOption) *StepMobile {
|
|
action := uixt.MobileAction{
|
|
Method: uixt.ACTION_Swipe,
|
|
Params: "down",
|
|
}
|
|
for _, option := range options {
|
|
option(&action)
|
|
}
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, action)
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
func (s *StepMobile) SwipeLeft(options ...uixt.ActionOption) *StepMobile {
|
|
action := uixt.MobileAction{
|
|
Method: uixt.ACTION_Swipe,
|
|
Params: "left",
|
|
}
|
|
for _, option := range options {
|
|
option(&action)
|
|
}
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, action)
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
func (s *StepMobile) SwipeRight(options ...uixt.ActionOption) *StepMobile {
|
|
action := uixt.MobileAction{
|
|
Method: uixt.ACTION_Swipe,
|
|
Params: "right",
|
|
}
|
|
for _, option := range options {
|
|
option(&action)
|
|
}
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, action)
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
func (s *StepMobile) SwipeToTapApp(appName string, options ...uixt.ActionOption) *StepMobile {
|
|
action := uixt.MobileAction{
|
|
Method: uixt.ACTION_SwipeToTapApp,
|
|
Params: appName,
|
|
}
|
|
for _, option := range options {
|
|
option(&action)
|
|
}
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, action)
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
func (s *StepMobile) SwipeToTapText(text string, options ...uixt.ActionOption) *StepMobile {
|
|
action := uixt.MobileAction{
|
|
Method: uixt.ACTION_SwipeToTapText,
|
|
Params: text,
|
|
}
|
|
for _, option := range options {
|
|
option(&action)
|
|
}
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, action)
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
func (s *StepMobile) SwipeToTapTexts(texts interface{}, options ...uixt.ActionOption) *StepMobile {
|
|
action := uixt.MobileAction{
|
|
Method: uixt.ACTION_SwipeToTapTexts,
|
|
Params: texts,
|
|
}
|
|
for _, option := range options {
|
|
option(&action)
|
|
}
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, action)
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
func (s *StepMobile) Input(text string, options ...uixt.ActionOption) *StepMobile {
|
|
action := uixt.MobileAction{
|
|
Method: uixt.ACTION_Input,
|
|
Params: text,
|
|
}
|
|
for _, option := range options {
|
|
option(&action)
|
|
}
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, action)
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
// Sleep specify sleep seconds after last action
|
|
func (s *StepMobile) Sleep(n float64) *StepMobile {
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, uixt.MobileAction{
|
|
Method: uixt.CtlSleep,
|
|
Params: n,
|
|
})
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
// SleepRandom specify random sleeping seconds after last action
|
|
// params have two different kinds:
|
|
// 1. [min, max] : min and max are float64 time range boudaries
|
|
// 2. [min1, max1, weight1, min2, max2, weight2, ...] : weight is the probability of the time range
|
|
func (s *StepMobile) SleepRandom(params ...float64) *StepMobile {
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, uixt.MobileAction{
|
|
Method: uixt.CtlSleepRandom,
|
|
Params: params,
|
|
})
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
func (s *StepMobile) ScreenShot() *StepMobile {
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, uixt.MobileAction{
|
|
Method: uixt.CtlScreenShot,
|
|
Params: nil,
|
|
})
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
func (s *StepMobile) StartCamera() *StepMobile {
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, uixt.MobileAction{
|
|
Method: uixt.CtlStartCamera,
|
|
Params: nil,
|
|
})
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
func (s *StepMobile) StopCamera() *StepMobile {
|
|
s.mobileStep().Actions = append(s.mobileStep().Actions, uixt.MobileAction{
|
|
Method: uixt.CtlStopCamera,
|
|
Params: nil,
|
|
})
|
|
return &StepMobile{step: s.step}
|
|
}
|
|
|
|
// Validate switches to step validation.
|
|
func (s *StepMobile) Validate() *StepMobileUIValidation {
|
|
return &StepMobileUIValidation{
|
|
step: s.step,
|
|
}
|
|
}
|
|
|
|
func (s *StepMobile) Name() string {
|
|
return s.step.Name
|
|
}
|
|
|
|
func (s *StepMobile) Type() StepType {
|
|
if s.step.Android != nil {
|
|
return stepTypeAndroid
|
|
}
|
|
return stepTypeIOS
|
|
}
|
|
|
|
func (s *StepMobile) Struct() *TStep {
|
|
return s.step
|
|
}
|
|
|
|
func (s *StepMobile) Run(r *SessionRunner) (*StepResult, error) {
|
|
return runStepMobileUI(r, s.step)
|
|
}
|
|
|
|
// StepMobileUIValidation implements IStep interface.
|
|
type StepMobileUIValidation struct {
|
|
step *TStep
|
|
}
|
|
|
|
func (s *StepMobileUIValidation) AssertNameExists(expectedName string, msg ...string) *StepMobileUIValidation {
|
|
v := Validator{
|
|
Check: uixt.SelectorName,
|
|
Assert: uixt.AssertionExists,
|
|
Expect: expectedName,
|
|
}
|
|
if len(msg) > 0 {
|
|
v.Message = msg[0]
|
|
} else {
|
|
v.Message = fmt.Sprintf("attribute name [%s] not found", expectedName)
|
|
}
|
|
s.step.Validators = append(s.step.Validators, v)
|
|
return s
|
|
}
|
|
|
|
func (s *StepMobileUIValidation) AssertNameNotExists(expectedName string, msg ...string) *StepMobileUIValidation {
|
|
v := Validator{
|
|
Check: uixt.SelectorName,
|
|
Assert: uixt.AssertionNotExists,
|
|
Expect: expectedName,
|
|
}
|
|
if len(msg) > 0 {
|
|
v.Message = msg[0]
|
|
} else {
|
|
v.Message = fmt.Sprintf("attribute name [%s] should not exist", expectedName)
|
|
}
|
|
s.step.Validators = append(s.step.Validators, v)
|
|
return s
|
|
}
|
|
|
|
func (s *StepMobileUIValidation) AssertLabelExists(expectedLabel string, msg ...string) *StepMobileUIValidation {
|
|
v := Validator{
|
|
Check: uixt.SelectorLabel,
|
|
Assert: uixt.AssertionExists,
|
|
Expect: expectedLabel,
|
|
}
|
|
if len(msg) > 0 {
|
|
v.Message = msg[0]
|
|
} else {
|
|
v.Message = fmt.Sprintf("attribute label [%s] not found", expectedLabel)
|
|
}
|
|
s.step.Validators = append(s.step.Validators, v)
|
|
return s
|
|
}
|
|
|
|
func (s *StepMobileUIValidation) AssertLabelNotExists(expectedLabel string, msg ...string) *StepMobileUIValidation {
|
|
v := Validator{
|
|
Check: uixt.SelectorLabel,
|
|
Assert: uixt.AssertionNotExists,
|
|
Expect: expectedLabel,
|
|
}
|
|
if len(msg) > 0 {
|
|
v.Message = msg[0]
|
|
} else {
|
|
v.Message = fmt.Sprintf("attribute label [%s] should not exist", expectedLabel)
|
|
}
|
|
s.step.Validators = append(s.step.Validators, v)
|
|
return s
|
|
}
|
|
|
|
func (s *StepMobileUIValidation) AssertOCRExists(expectedText string, msg ...string) *StepMobileUIValidation {
|
|
v := Validator{
|
|
Check: uixt.SelectorOCR,
|
|
Assert: uixt.AssertionExists,
|
|
Expect: expectedText,
|
|
}
|
|
if len(msg) > 0 {
|
|
v.Message = msg[0]
|
|
} else {
|
|
v.Message = fmt.Sprintf("ocr text [%s] not found", expectedText)
|
|
}
|
|
s.step.Validators = append(s.step.Validators, v)
|
|
return s
|
|
}
|
|
|
|
func (s *StepMobileUIValidation) AssertOCRNotExists(expectedText string, msg ...string) *StepMobileUIValidation {
|
|
v := Validator{
|
|
Check: uixt.SelectorOCR,
|
|
Assert: uixt.AssertionNotExists,
|
|
Expect: expectedText,
|
|
}
|
|
if len(msg) > 0 {
|
|
v.Message = msg[0]
|
|
} else {
|
|
v.Message = fmt.Sprintf("ocr text [%s] should not exist", expectedText)
|
|
}
|
|
s.step.Validators = append(s.step.Validators, v)
|
|
return s
|
|
}
|
|
|
|
func (s *StepMobileUIValidation) AssertImageExists(expectedImagePath string, msg ...string) *StepMobileUIValidation {
|
|
v := Validator{
|
|
Check: uixt.SelectorImage,
|
|
Assert: uixt.AssertionExists,
|
|
Expect: expectedImagePath,
|
|
}
|
|
if len(msg) > 0 {
|
|
v.Message = msg[0]
|
|
} else {
|
|
v.Message = fmt.Sprintf("cv image [%s] not found", expectedImagePath)
|
|
}
|
|
s.step.Validators = append(s.step.Validators, v)
|
|
return s
|
|
}
|
|
|
|
func (s *StepMobileUIValidation) AssertImageNotExists(expectedImagePath string, msg ...string) *StepMobileUIValidation {
|
|
v := Validator{
|
|
Check: uixt.SelectorImage,
|
|
Assert: uixt.AssertionNotExists,
|
|
Expect: expectedImagePath,
|
|
}
|
|
if len(msg) > 0 {
|
|
v.Message = msg[0]
|
|
} else {
|
|
v.Message = fmt.Sprintf("cv image [%s] should not exist", expectedImagePath)
|
|
}
|
|
s.step.Validators = append(s.step.Validators, v)
|
|
return s
|
|
}
|
|
|
|
func (s *StepMobileUIValidation) AssertAppInForeground(packageName string, msg ...string) *StepMobileUIValidation {
|
|
v := Validator{
|
|
Check: uixt.SelectorForegroundApp,
|
|
Assert: uixt.AssertionEqual,
|
|
Expect: packageName,
|
|
}
|
|
if len(msg) > 0 {
|
|
v.Message = msg[0]
|
|
} else {
|
|
v.Message = fmt.Sprintf("app [%s] should be in foreground", packageName)
|
|
}
|
|
s.step.Validators = append(s.step.Validators, v)
|
|
return s
|
|
}
|
|
|
|
func (s *StepMobileUIValidation) AssertAppNotInForeground(packageName string, msg ...string) *StepMobileUIValidation {
|
|
v := Validator{
|
|
Check: uixt.SelectorForegroundApp,
|
|
Assert: uixt.AssertionNotEqual,
|
|
Expect: packageName,
|
|
}
|
|
if len(msg) > 0 {
|
|
v.Message = msg[0]
|
|
} else {
|
|
v.Message = fmt.Sprintf("app [%s] should not be in foreground", packageName)
|
|
}
|
|
s.step.Validators = append(s.step.Validators, v)
|
|
return s
|
|
}
|
|
|
|
func (s *StepMobileUIValidation) Name() string {
|
|
return s.step.Name
|
|
}
|
|
|
|
func (s *StepMobileUIValidation) Type() StepType {
|
|
return stepTypeIOS
|
|
}
|
|
|
|
func (s *StepMobileUIValidation) Struct() *TStep {
|
|
return s.step
|
|
}
|
|
|
|
func (s *StepMobileUIValidation) Run(r *SessionRunner) (*StepResult, error) {
|
|
return runStepMobileUI(r, s.step)
|
|
}
|
|
|
|
func (r *HRPRunner) initUIClient(uuid string, osType string) (client *uixt.DriverExt, err error) {
|
|
// avoid duplicate init
|
|
if uuid == "" && len(r.uiClients) > 0 {
|
|
for _, v := range r.uiClients {
|
|
return v, nil
|
|
}
|
|
}
|
|
|
|
// avoid duplicate init
|
|
if uuid != "" {
|
|
if client, ok := r.uiClients[uuid]; ok {
|
|
return client, nil
|
|
}
|
|
}
|
|
|
|
var device uixt.Device
|
|
if osType == "ios" {
|
|
device, err = uixt.NewIOSDevice(uixt.WithUDID(uuid))
|
|
} else {
|
|
device, err = uixt.NewAndroidDevice(uixt.WithSerialNumber(uuid))
|
|
}
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "init %s device failed", osType)
|
|
}
|
|
|
|
client, err = device.NewDriver(nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// cache wda client
|
|
if r.uiClients == nil {
|
|
r.uiClients = make(map[string]*uixt.DriverExt)
|
|
}
|
|
r.uiClients[client.Device.UUID()] = client
|
|
|
|
return client, nil
|
|
}
|
|
|
|
func runStepMobileUI(s *SessionRunner, step *TStep) (stepResult *StepResult, err error) {
|
|
var osType string
|
|
var mobileStep *MobileStep
|
|
if step.IOS != nil {
|
|
// ios step
|
|
osType = "ios"
|
|
mobileStep = step.IOS
|
|
} else {
|
|
// android step
|
|
osType = "android"
|
|
mobileStep = step.Android
|
|
}
|
|
|
|
stepResult = &StepResult{
|
|
Name: step.Name,
|
|
StepType: StepType(osType),
|
|
Success: false,
|
|
ContentSize: 0,
|
|
}
|
|
|
|
// merge step variables with session variables
|
|
stepVariables, err := s.ParseStepVariables(step.Variables)
|
|
if err != nil {
|
|
err = errors.Wrap(err, "parse step variables failed")
|
|
return
|
|
}
|
|
|
|
// init wda/uia driver
|
|
uiDriver, err := s.caseRunner.hrpRunner.initUIClient(mobileStep.Serial, osType)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
defer func() {
|
|
attachments := make(map[string]interface{})
|
|
if err != nil {
|
|
attachments["error"] = err.Error()
|
|
|
|
// check if app is in foreground
|
|
packageName := uiDriver.Driver.GetLastLaunchedApp()
|
|
yes, err2 := uiDriver.Driver.IsAppInForeground(packageName)
|
|
if packageName != "" && (!yes || err2 != nil) {
|
|
log.Error().Err(err2).Str("packageName", packageName).Msg("app is not in foreground")
|
|
err = errors.Wrap(code.MobileUIAppNotInForegroundError, err.Error())
|
|
}
|
|
}
|
|
|
|
// take screenshot after each step
|
|
_, err := uiDriver.TakeScreenShot(
|
|
builtin.GenNameWithTimestamp("step_%d_") + step.Name)
|
|
if err != nil {
|
|
log.Error().Err(err).Str("step", step.Name).Msg("take screenshot failed on step finished")
|
|
}
|
|
|
|
// save attachments
|
|
attachments["screenshots"] = uiDriver.GetScreenShots()
|
|
stepResult.Attachments = attachments
|
|
}()
|
|
|
|
// prepare actions
|
|
var actions []uixt.MobileAction
|
|
if mobileStep.Actions == nil {
|
|
actions = []uixt.MobileAction{
|
|
{
|
|
Method: mobileStep.Method,
|
|
Params: mobileStep.Params,
|
|
},
|
|
}
|
|
} else {
|
|
actions = mobileStep.Actions
|
|
}
|
|
|
|
// run actions
|
|
for _, action := range actions {
|
|
if action.Params, err = s.caseRunner.parser.Parse(action.Params, stepVariables); err != nil {
|
|
if !code.IsErrorPredefined(err) {
|
|
err = errors.Wrap(code.ParseError,
|
|
fmt.Sprintf("parse action params failed: %v", err))
|
|
}
|
|
return stepResult, err
|
|
}
|
|
if err := uiDriver.DoAction(action); err != nil {
|
|
if !code.IsErrorPredefined(err) {
|
|
err = errors.Wrap(code.MobileUIDriverError, err.Error())
|
|
}
|
|
return stepResult, err
|
|
}
|
|
}
|
|
|
|
// validate
|
|
validateResults, err := validateUI(uiDriver, step.Validators)
|
|
if err != nil {
|
|
if !code.IsErrorPredefined(err) {
|
|
err = errors.Wrap(code.MobileUIValidationError, err.Error())
|
|
}
|
|
return
|
|
}
|
|
sessionData := newSessionData()
|
|
sessionData.Validators = validateResults
|
|
stepResult.Data = sessionData
|
|
stepResult.Success = true
|
|
return stepResult, nil
|
|
}
|