fix: WDA tests

This commit is contained in:
lilong.129
2025-02-17 22:02:12 +08:00
parent 3ae314d0ba
commit e985697227
12 changed files with 298 additions and 327 deletions

View File

@@ -316,8 +316,3 @@ func (p PointF) IsIdentical(p2 PointF) bool {
// set the coordinate precision to 1 pixel
return math.Abs(p.X-p2.X) < 1 && math.Abs(p.Y-p2.Y) < 1
}
type Screen struct {
StatusBarSize types.Size `json:"statusBarSize"`
Scale float64 `json:"scale"`
}

View File

@@ -239,7 +239,7 @@ func (ud *UIA2Driver) TapAbsXY(x, y float64, opts ...option.ActionOption) error
duration := 100.0
if actionOptions.PressDuration > 0 {
duration = actionOptions.PressDuration * 1000
duration = actionOptions.PressDuration * 1000 // convert to ms
}
data := map[string]interface{}{
"actions": []interface{}{
@@ -256,9 +256,7 @@ func (ud *UIA2Driver) TapAbsXY(x, y float64, opts ...option.ActionOption) error
},
},
}
// update data options in post data for extra uiautomator configurations
actionOptions.UpdateData(data)
option.MergeOptions(data, opts...)
_, err := ud.Session.POST(data, "/session", ud.Session.ID, "actions/tap")
return err
@@ -301,9 +299,7 @@ func (ud *UIA2Driver) Drag(fromX, fromY, toX, toY float64, opts ...option.Action
"endX": toX,
"endY": toY,
}
// update data options in post data for extra uiautomator configurations
actionOptions.UpdateData(data)
option.MergeOptions(data, opts...)
// register(postHandler, new Drag("/wd/hub/session/:sessionId/touch/drag"))
_, err = ud.Session.POST(data, "/session", ud.Session.ID, "touch/drag")
@@ -328,7 +324,7 @@ func (ud *UIA2Driver) Swipe(fromX, fromY, toX, toY float64, opts ...option.Actio
duration := 200.0
if actionOptions.PressDuration > 0 {
duration = actionOptions.PressDuration * 1000
duration = actionOptions.PressDuration * 1000 // ms
}
data := map[string]interface{}{
"actions": []interface{}{
@@ -345,9 +341,7 @@ func (ud *UIA2Driver) Swipe(fromX, fromY, toX, toY float64, opts ...option.Actio
},
},
}
// update data options in post data for extra uiautomator configurations
actionOptions.UpdateData(data)
option.MergeOptions(data, opts...)
_, err = ud.Session.POST(data, "/session", ud.Session.ID, "actions/swipe")
return err
@@ -400,18 +394,16 @@ func (ud *UIA2Driver) GetPasteboard(contentType types.PasteboardType) (raw *byte
func (ud *UIA2Driver) Input(text string, opts ...option.ActionOption) (err error) {
// register(postHandler, new SendKeysToElement("/wd/hub/session/:sessionId/keys"))
// https://github.com/appium/appium-uiautomator2-server/blob/master/app/src/main/java/io/appium/uiautomator2/handler/SendKeysToElement.java#L76-L85
actionOptions := option.NewActionOptions(opts...)
err = ud.SendUnicodeKeys(text, opts...)
if err != nil {
data := map[string]interface{}{
"text": text,
}
// new data options in post data for extra uiautomator configurations
actionOptions.UpdateData(data)
_, err = ud.Session.POST(data, "/session", ud.Session.ID, "/keys")
if err == nil {
return nil
}
data := map[string]interface{}{
"text": text,
}
option.MergeOptions(data, opts...)
_, err = ud.Session.POST(data, "/session", ud.Session.ID, "/keys")
return
}
@@ -446,7 +438,6 @@ func (ud *UIA2Driver) SendUnicodeKeys(text string, opts ...option.ActionOption)
}
func (ud *UIA2Driver) SendActionKey(text string, opts ...option.ActionOption) (err error) {
actionOptions := option.NewActionOptions(opts...)
var actions []interface{}
for i, c := range text {
actions = append(actions, map[string]interface{}{"type": "keyDown", "value": string(c)},
@@ -465,9 +456,8 @@ func (ud *UIA2Driver) SendActionKey(text string, opts ...option.ActionOption) (e
},
},
}
option.MergeOptions(data, opts...)
// new data options in post data for extra uiautomator configurations
actionOptions.UpdateData(data)
_, err = ud.Session.POST(data, "/session", ud.Session.ID, "/actions/keys")
return
}

View File

@@ -14,7 +14,10 @@ import (
"github.com/httprunner/httprunner/v5/pkg/uixt/types"
)
var driverExt *XTDriver
var (
driver IDriver
driverExt *XTDriver
)
func setupAndroidAdbDriver(t *testing.T) {
device, err := NewAndroidDevice()

View File

@@ -176,8 +176,7 @@ func (dExt *XTDriver) DoAction(action MobileAction) (err error) {
}
case ACTION_GetSource:
if packageName, ok := action.Params.(string); ok {
source := option.NewSourceOption().WithProcessName(packageName)
_, err = dExt.Source(source)
_, err = dExt.Source(option.WithProcessName(packageName))
if err != nil {
return errors.Wrap(err, "failed to set ime")
}

View File

@@ -6,6 +6,7 @@ import (
"image"
"image/gif"
"image/jpeg"
"image/png"
_ "image/png"
"os"
"path/filepath"
@@ -214,14 +215,14 @@ func saveScreenShot(raw *bytes.Buffer, fileName string) (string, error) {
// compress image and save to file
switch format {
case "jpeg", "png":
case "jpeg":
jpegOptions := &jpeg.Options{Quality: 95}
err = jpeg.Encode(file, img, jpegOptions)
// case "png":
// encoder := png.Encoder{
// CompressionLevel: png.BestCompression,
// }
// err = encoder.Encode(file, img)
case "png":
encoder := png.Encoder{
CompressionLevel: png.BestCompression,
}
err = encoder.Encode(file, img)
case "gif":
gifOptions := &gif.Options{
NumColors: 256,

View File

@@ -2,6 +2,7 @@ package uixt
import (
"fmt"
"math"
"math/rand/v2"
"time"
@@ -43,8 +44,8 @@ func convertToAbsolutePoint(driver IDriver, x, y float64) (absX, absY float64, e
return
}
absX = float64(windowSize.Width) * x
absY = float64(windowSize.Height) * y
absX = math.Round(float64(windowSize.Width)*x*10) / 10
absY = math.Round(float64(windowSize.Height)*y*10) / 10
return
}

View File

@@ -9,7 +9,6 @@ import (
"math"
"net"
"net/http"
"net/url"
"os"
"os/exec"
"path/filepath"
@@ -25,7 +24,6 @@ import (
"github.com/httprunner/httprunner/v5/internal/builtin"
"github.com/httprunner/httprunner/v5/internal/config"
"github.com/httprunner/httprunner/v5/internal/json"
"github.com/httprunner/httprunner/v5/pkg/uixt/ai"
"github.com/httprunner/httprunner/v5/pkg/uixt/option"
"github.com/httprunner/httprunner/v5/pkg/uixt/types"
)
@@ -37,58 +35,22 @@ func NewWDADriver(device *IOSDevice) (*WDADriver, error) {
Session: NewDriverSession(),
}
var err error
// forward local port to device
var localPort int
localPort, err = strconv.Atoi(os.Getenv("WDA_LOCAL_PORT"))
if err != nil {
localPort, err = builtin.GetFreePort()
if err != nil {
return nil, errors.Wrap(code.DeviceHTTPDriverError,
fmt.Sprintf("get free port failed: %v", err))
}
if err = device.Forward(localPort, device.Options.WDAPort); err != nil {
return nil, errors.Wrap(code.DeviceHTTPDriverError,
fmt.Sprintf("forward tcp port failed: %v", err))
}
} else {
log.Info().Int("WDA_LOCAL_PORT", localPort).Msg("reuse WDA local port")
}
var localMjpegPort int
localMjpegPort, err = strconv.Atoi(os.Getenv("WDA_LOCAL_MJPEG_PORT"))
if err != nil {
localMjpegPort, err = builtin.GetFreePort()
if err != nil {
return nil, errors.Wrap(code.DeviceHTTPDriverError,
fmt.Sprintf("get free port failed: %v", err))
}
if err = device.Forward(localMjpegPort, device.Options.WDAMjpegPort); err != nil {
return nil, errors.Wrap(code.DeviceHTTPDriverError,
fmt.Sprintf("forward tcp port failed: %v", err))
}
} else {
log.Info().Int("WDA_LOCAL_MJPEG_PORT", localMjpegPort).
Msg("reuse WDA local mjpeg port")
}
host := "localhost"
localPort, err := driver.getLocalPort()
if err != nil {
return nil, err
}
driver.Session.SetBaseURL(fmt.Sprintf("http://%s:%d", host, localPort))
if err = driver.initMjpegClient(); err != nil {
return nil, err
}
// create new session
if err = driver.InitSession(nil); err != nil {
return nil, errors.Wrap(code.DeviceHTTPDriverError, err.Error())
}
if driver.mjpegHTTPConn, err = net.Dial(
"tcp",
fmt.Sprintf("%s:%d", host, localMjpegPort),
); err != nil {
return nil, errors.Wrap(code.DeviceHTTPDriverError, err.Error())
}
driver.mjpegClient = NewHTTPClientWithConnection(driver.mjpegHTTPConn, 30*time.Second)
driver.mjpegUrl = fmt.Sprintf("%s:%d", host, localMjpegPort)
// init WDA scale
if driver.scale, err = driver.Scale(); err != nil {
return nil, err
@@ -110,6 +72,61 @@ type WDADriver struct {
mjpegUrl string
}
func (wd *WDADriver) getLocalPort() (int, error) {
localPort, err := strconv.Atoi(os.Getenv("WDA_LOCAL_PORT"))
if err != nil {
localPort, err = builtin.GetFreePort()
if err != nil {
return 0, errors.Wrap(code.DeviceHTTPDriverError,
fmt.Sprintf("get free port failed: %v", err))
}
// forward local port to device
if err = wd.Device.Forward(localPort, wd.Device.Options.WDAPort); err != nil {
return 0, errors.Wrap(code.DeviceHTTPDriverError,
fmt.Sprintf("forward tcp port failed: %v", err))
}
} else {
log.Info().Int("WDA_LOCAL_PORT", localPort).Msg("reuse WDA local port")
}
return localPort, nil
}
func (wd *WDADriver) getMjpegLocalPort() (int, error) {
localMjpegPort, err := strconv.Atoi(os.Getenv("WDA_LOCAL_MJPEG_PORT"))
if err != nil {
localMjpegPort, err = builtin.GetFreePort()
if err != nil {
return 0, errors.Wrap(code.DeviceHTTPDriverError,
fmt.Sprintf("get free port failed: %v", err))
}
if err = wd.Device.Forward(localMjpegPort, wd.Device.Options.WDAMjpegPort); err != nil {
return 0, errors.Wrap(code.DeviceHTTPDriverError,
fmt.Sprintf("forward tcp port failed: %v", err))
}
} else {
log.Info().Int("WDA_LOCAL_MJPEG_PORT", localMjpegPort).
Msg("reuse WDA local mjpeg port")
}
return localMjpegPort, nil
}
func (wd *WDADriver) initMjpegClient() error {
host := "localhost"
localMjpegPort, err := wd.getMjpegLocalPort()
if err != nil {
return err
}
if wd.mjpegHTTPConn, err = net.Dial(
"tcp",
fmt.Sprintf("%s:%d", host, localMjpegPort),
); err != nil {
return errors.Wrap(code.DeviceHTTPDriverError, err.Error())
}
wd.mjpegClient = NewHTTPClientWithConnection(wd.mjpegHTTPConn, 30*time.Second)
wd.mjpegUrl = fmt.Sprintf("http://%s:%d", host, localMjpegPort)
return nil
}
func (wd *WDADriver) GetMjpegClient() *http.Client {
return wd.mjpegClient
}
@@ -145,6 +162,8 @@ func (wd *WDADriver) DeleteSession() (err error) {
// [[FBRoute DELETE:@""] respondWithTarget:self action:@selector(handleDeleteSession:)]
_, err = wd.Session.DELETE("/session", wd.Session.ID)
wd.Session.ID = ""
wd.Session.client.CloseIdleConnections()
return
}
@@ -250,15 +269,20 @@ func (wd *WDADriver) Scale() (float64, error) {
return screen.Scale, nil
}
func (wd *WDADriver) Screen() (screen ai.Screen, err error) {
type Screen struct {
StatusBarSize types.Size `json:"statusBarSize"`
Scale float64 `json:"scale"`
}
func (wd *WDADriver) Screen() (screen Screen, err error) {
// [[FBRoute GET:@"/wda/screen"] respondWithTarget:self action:@selector(handleGetScreen:)]
var rawResp DriverRawResponse
if rawResp, err = wd.Session.GET("/session", wd.Session.ID, "/wda/screen"); err != nil {
return ai.Screen{}, err
return Screen{}, err
}
reply := new(struct{ Value struct{ ai.Screen } })
reply := new(struct{ Value struct{ Screen } })
if err = json.Unmarshal(rawResp, reply); err != nil {
return ai.Screen{}, err
return Screen{}, err
}
screen = reply.Value.Screen
return
@@ -525,6 +549,8 @@ func (wd *WDADriver) TapAbsXY(x, y float64, opts ...option.ActionOption) error {
"x": wd.toScale(x),
"y": wd.toScale(y),
}
option.MergeOptions(data, opts...)
_, err := wd.Session.POST(data, "/session", wd.Session.ID, "/wda/tap/0")
return err
}
@@ -546,10 +572,11 @@ func (wd *WDADriver) DoubleTapXY(x, y float64, opts ...option.ActionOption) erro
return err
}
// FIXME: hold not work
func (wd *WDADriver) TouchAndHold(x, y float64, opts ...option.ActionOption) (err error) {
actionOptions := option.NewActionOptions(opts...)
if actionOptions.Duration == 0 {
opts = append(opts, option.WithDuration(1))
opts = append(opts, option.WithPressDuration(1))
}
return wd.TapXY(x, y, opts...)
}
@@ -574,12 +601,7 @@ func (wd *WDADriver) Drag(fromX, fromY, toX, toY float64, opts ...option.ActionO
"toX": math.Round(toX*10) / 10,
"toY": math.Round(toY*10) / 10,
}
if actionOptions.PressDuration > 0 {
data["pressDuration"] = actionOptions.PressDuration
}
// update data options in post data for extra WDA configurations
actionOptions.UpdateData(data)
option.MergeOptions(data, opts...)
// wda 43 version
_, err = wd.Session.POST(data, "/session", wd.Session.ID, "/wda/dragfromtoforduration")
// _, err = wd.Session.POST(data, "/session", wd.Session.ID, "/wda/drag")
@@ -619,12 +641,8 @@ func (wd *WDADriver) SetIme(ime string) error {
func (wd *WDADriver) Input(text string, opts ...option.ActionOption) (err error) {
// [[FBRoute POST:@"/wda/keys"] respondWithTarget:self action:@selector(handleKeys:)]
actionOptions := option.NewActionOptions(opts...)
data := map[string]interface{}{"value": strings.Split(text, "")}
// new data options in post data for extra WDA configurations
actionOptions.UpdateData(data)
option.MergeOptions(data, opts...)
_, err = wd.Session.POST(data, "/session", wd.Session.ID, "/wda/keys")
return
}
@@ -633,12 +651,8 @@ func (wd *WDADriver) Backspace(count int, opts ...option.ActionOption) (err erro
if count == 0 {
return nil
}
actionOptions := option.NewActionOptions(opts...)
data := map[string]interface{}{"count": count}
// new data options in post data for extra WDA configurations
actionOptions.UpdateData(data)
option.MergeOptions(data, opts...)
_, err = wd.Session.POST(data, "/gtf/interaction/input/backspace")
return
}
@@ -720,39 +734,29 @@ func (wd *WDADriver) SetRotation(rotation types.Rotation) (err error) {
func (wd *WDADriver) Source(srcOpt ...option.SourceOption) (source string, err error) {
// [[FBRoute GET:@"/source"] respondWithTarget:self action:@selector(handleGetSourceCommand:)]
// [[FBRoute GET:@"/source"].withoutSession
urlStr, err := wd.Session.concatURL("/session", wd.Session.ID)
if err != nil {
return "", err
}
tmp, _ := url.Parse(urlStr)
toJsonRaw := false
if len(srcOpt) != 0 {
q := tmp.Query()
for k, val := range srcOpt[0] {
v := val.(string)
q.Set(k, v)
if k == "format" && v == "json" {
toJsonRaw = true
}
}
tmp.RawQuery = q.Encode()
}
urlStr, err = wd.Session.concatURL(tmp.Path, "/source")
if err != nil {
return "", err
// urlStr, err := wd.Session.concatURL("/session", wd.Session.ID)
// if err != nil {
// return "", err
// }
options := option.NewSourceOptions(srcOpt...)
query := options.Query()
if len(query) > 0 {
query = "?" + query
}
var rawResp DriverRawResponse
if rawResp, err = wd.Session.GET(http.MethodGet, urlStr); err != nil {
return "", nil
if rawResp, err = wd.Session.GET("/source" + query); err != nil {
return "", err
}
if toJsonRaw {
// json format
if options.Format == option.SourceFormatJSON {
var jr builtinJSON.RawMessage
if jr, err = rawResp.ValueConvertToJsonRawMessage(); err != nil {
return "", err
}
return string(jr), nil
}
// xml/description format
if source, err = rawResp.ValueConvertToString(); err != nil {
return "", err
}
@@ -850,7 +854,7 @@ func (wd *WDADriver) ScreenRecord(duration time.Duration) (videoPath string, err
"-f", "mjpeg",
"-y",
"-r", "10",
"-i", "http://"+wd.mjpegUrl,
"-i", wd.mjpegUrl,
"-c:v", "libx264",
"-vf", "pad=width=ceil(iw/2)*2:height=ceil(ih/2)*2",
fileName,
@@ -936,7 +940,5 @@ func (wd *WDADriver) Setup() error {
}
func (wd *WDADriver) TearDown() error {
wd.mjpegClient.CloseIdleConnections()
wd.Session.client.CloseIdleConnections()
return nil
return wd.DeleteSession()
}

View File

@@ -14,13 +14,7 @@ import (
"github.com/httprunner/httprunner/v5/pkg/uixt/types"
)
var (
bundleId = "com.apple.Preferences"
driver IDriver
iOSDriverExt *XTDriver
)
func setup(t *testing.T) {
func setupWDADriverExt(t *testing.T) *XTDriver {
device, err := NewIOSDevice(
option.WithWDAPort(8700),
option.WithWDAMjpegPort(8800),
@@ -28,21 +22,16 @@ func setup(t *testing.T) {
if err != nil {
t.Fatal(err)
}
driver, err = device.NewDriver()
driver, err := device.NewDriver()
if err != nil {
t.Fatal(err)
}
iOSDriverExt = NewXTDriver(driver, ai.WithCVService(ai.CVServiceTypeVEDEM))
}
func TestViaUSB(t *testing.T) {
setup(t)
t.Log(driver.Status())
return NewXTDriver(driver, ai.WithCVService(ai.CVServiceTypeVEDEM))
}
func TestInstall(t *testing.T) {
setup(t)
err := iOSDriverExt.GetDevice().Install("xxx.ipa",
driver := setupWDADriverExt(t)
err := driver.GetDevice().Install("xxx.ipa",
option.WithRetryTimes(5))
log.Error().Err(err)
if err != nil {
@@ -84,28 +73,22 @@ func TestIOSDevice_GetPackageInfo(t *testing.T) {
checkErr(t, err)
appInfo, err := device.GetPackageInfo("com.ss.iphone.ugc.Aweme")
checkErr(t, err)
t.Log(appInfo)
}
func TestNewUSBDriver(t *testing.T) {
setup(t)
// t.Log(driver.IsWdaHealthy())
t.Logf("%+v", appInfo)
}
func TestDriver_DeviceScaleRatio(t *testing.T) {
setup(t)
driver := setupWDADriverExt(t)
scaleRatio, err := driver.(*WDADriver).Scale()
scaleRatio, err := driver.IDriver.(*WDADriver).Scale()
if err != nil {
t.Fatal(err)
}
t.Log(scaleRatio)
t.Logf("%+v", scaleRatio)
}
func Test_remoteWD_DeleteSession(t *testing.T) {
setup(t)
driver := setupWDADriverExt(t)
err := driver.DeleteSession()
if err != nil {
@@ -114,26 +97,26 @@ func Test_remoteWD_DeleteSession(t *testing.T) {
}
func Test_remoteWD_HealthCheck(t *testing.T) {
setup(t)
driver := setupWDADriverExt(t)
err := driver.(*WDADriver).HealthCheck()
err := driver.IDriver.(*WDADriver).HealthCheck()
if err != nil {
t.Fatal(err)
}
}
func Test_remoteWD_GetAppiumSettings(t *testing.T) {
setup(t)
driver := setupWDADriverExt(t)
settings, err := driver.(*WDADriver).GetAppiumSettings()
settings, err := driver.IDriver.(*WDADriver).GetAppiumSettings()
if err != nil {
t.Fatal(err)
}
t.Log(settings)
t.Logf("%+v", settings)
}
func Test_remoteWD_SetAppiumSettings(t *testing.T) {
setup(t)
driver := setupWDADriverExt(t)
const _acceptAlertButtonSelector = "**/XCUIElementTypeButton[`label IN {'允许','好','仅在使用应用期间','暂不'}`]"
const _dismissAlertButtonSelector = "**/XCUIElementTypeButton[`label IN {'不允许','暂不'}`]"
@@ -142,7 +125,7 @@ func Test_remoteWD_SetAppiumSettings(t *testing.T) {
value := _acceptAlertButtonSelector
// settings, err := driver.SetAppiumSettings(map[string]interface{}{"dismissAlertButtonSelector": "暂不"})
settings, err := driver.(*WDADriver).SetAppiumSettings(map[string]interface{}{key: value})
settings, err := driver.IDriver.(*WDADriver).SetAppiumSettings(map[string]interface{}{key: value})
if err != nil {
t.Fatal(err)
}
@@ -152,39 +135,31 @@ func Test_remoteWD_SetAppiumSettings(t *testing.T) {
}
func Test_remoteWD_IsWdaHealthy(t *testing.T) {
setup(t)
driver := setupWDADriverExt(t)
healthy, err := driver.(*WDADriver).IsHealthy()
healthy, err := driver.IDriver.(*WDADriver).IsHealthy()
if err != nil {
t.Fatal(err)
}
if healthy == false {
t.Fatal("healthy =", healthy)
if !healthy {
t.Fatal("assert healthy failed")
}
}
// func Test_remoteWD_WdaShutdown(t *testing.T) {
// setup(t)
//
// if err := driver.WdaShutdown(); err != nil {
// t.Fatal(err)
// }
// }
func Test_remoteWD_Status(t *testing.T) {
setup(t)
driver := setupWDADriverExt(t)
status, err := driver.Status()
if err != nil {
t.Fatal(err)
}
if status.Ready == false {
t.Fatal("deviceStatus =", status)
if !status.Ready {
t.Fatal("assert device status failed")
}
}
func Test_remoteWD_DeviceInfo(t *testing.T) {
setup(t)
driver := setupWDADriverExt(t)
info, err := driver.DeviceInfo()
if err != nil {
@@ -196,7 +171,7 @@ func Test_remoteWD_DeviceInfo(t *testing.T) {
}
func Test_remoteWD_BatteryInfo(t *testing.T) {
setup(t)
driver := setupWDADriverExt(t)
batteryInfo, err := driver.BatteryInfo()
if err != nil {
@@ -206,7 +181,7 @@ func Test_remoteWD_BatteryInfo(t *testing.T) {
}
func Test_remoteWD_WindowSize(t *testing.T) {
setup(t)
driver := setupWDADriverExt(t)
size, err := driver.WindowSize()
if err != nil {
@@ -216,9 +191,9 @@ func Test_remoteWD_WindowSize(t *testing.T) {
}
func Test_remoteWD_Screen(t *testing.T) {
setup(t)
driver := setupWDADriverExt(t)
screen, err := driver.(*WDADriver).Screen()
screen, err := driver.IDriver.(*WDADriver).Screen()
if err != nil {
t.Fatal(err)
}
@@ -226,7 +201,7 @@ func Test_remoteWD_Screen(t *testing.T) {
}
func Test_remoteWD_Homescreen(t *testing.T) {
setup(t)
driver := setupWDADriverExt(t)
err := driver.Home()
if err != nil {
@@ -234,174 +209,154 @@ func Test_remoteWD_Homescreen(t *testing.T) {
}
}
func Test_remoteWD_AppLaunch(t *testing.T) {
setup(t)
func Test_remoteWD_AppLaunchTerminate(t *testing.T) {
driver := setupWDADriverExt(t)
bundleId := "com.apple.Preferences"
err := driver.AppLaunch(bundleId)
// err := driver.AppLaunch(bundleId, NewAppLaunchOption().WithShouldWaitForQuiescence(true))
// err := driver.AppLaunch(bundleId, NewAppLaunchOption().WithArguments([]string{"-AppleLanguages", "(Russian)"}))
if err != nil {
t.Fatal(err)
}
}
time.Sleep(2 * time.Second)
func Test_remoteWD_AppTerminate(t *testing.T) {
setup(t)
_, err := driver.AppTerminate(bundleId)
_, err = driver.AppTerminate(bundleId)
if err != nil {
t.Fatal(err)
}
}
func Test_remoteWD_Tap(t *testing.T) {
setup(t)
driver := setupWDADriverExt(t)
err := driver.TapXY(200, 300)
err := driver.TapXY(0.2, 0.2)
if err != nil {
t.Fatal(err)
}
}
func Test_remoteWD_DoubleTap(t *testing.T) {
setup(t)
driver := setupWDADriverExt(t)
err := driver.DoubleTapXY(200, 300)
err := driver.DoubleTapXY(0.2, 0.2)
if err != nil {
t.Fatal(err)
}
}
func Test_remoteWD_TouchAndHold(t *testing.T) {
setup(t)
driver := setupWDADriverExt(t)
// err := driver.TouchAndHold(200, 300)
err := driver.TouchAndHold(200, 300)
err := driver.TouchAndHold(0.2, 0.2)
if err != nil {
t.Fatal(err)
}
}
func Test_remoteWD_Drag(t *testing.T) {
setup(t)
driver := setupWDADriverExt(t)
// err := driver.Drag(200, 300, 200, 500, WithDataPressDuration(0.5))
err := driver.Drag(200, 300, 200, 500,
option.WithPressDuration(2), option.WithDuration(3))
err := driver.Drag(0.8, 0.5, 0.2, 0.5,
option.WithDuration(0.5))
if err != nil {
t.Fatal(err)
}
}
func Test_Relative_Drag(t *testing.T) {
setup(t)
func Test_Relative_Swipe(t *testing.T) {
driver := setupWDADriverExt(t)
// err := driver.Drag(200, 300, 200, 500, WithDataPressDuration(0.5))
err := iOSDriverExt.Swipe(0.5, 0.7, 0.5, 0.5)
err := driver.Swipe(0.8, 0.5, 0.2, 0.5)
if err != nil {
t.Fatal(err)
}
}
func Test_remoteWD_SendKeys(t *testing.T) {
setup(t)
// driver.StartCaptureLog("hrp_wda_log")
err := driver.Input("test", option.WithIdentifier("test"))
// result, _ := driver.StopCaptureLog()
// err := driver.SendKeys("App Store", WithFrequency(3))
driver := setupWDADriverExt(t)
driver.StartCaptureLog("hrp_wda_log")
err := driver.Input("test中文", option.WithIdentifier("test"))
result, _ := driver.StopCaptureLog()
if err != nil {
t.Fatal(err)
}
// t.Log(result)
t.Log(result)
}
func Test_remoteWD_PressButton(t *testing.T) {
setup(t)
driver := setupWDADriverExt(t)
err := driver.(*WDADriver).PressButton(types.DeviceButtonVolumeUp)
err := driver.IDriver.(*WDADriver).PressButton(types.DeviceButtonVolumeUp)
if err != nil {
t.Fatal(err)
}
time.Sleep(time.Second * 1)
err = driver.(*WDADriver).PressButton(types.DeviceButtonVolumeDown)
err = driver.IDriver.(*WDADriver).PressButton(types.DeviceButtonVolumeDown)
if err != nil {
t.Fatal(err)
}
time.Sleep(time.Second * 1)
err = driver.(*WDADriver).PressButton(types.DeviceButtonHome)
err = driver.IDriver.(*WDADriver).PressButton(types.DeviceButtonHome)
if err != nil {
t.Fatal(err)
}
}
func Test_remoteWD_Screenshot(t *testing.T) {
setup(t)
driver := setupWDADriverExt(t)
// without save file
screenshot, err := driver.ScreenShot()
if err != nil {
t.Fatal(err)
}
_ = screenshot
// img, format, err := image.Decode(screenshot)
// if err != nil {
// t.Fatal(err)
// }
// userHomeDir, _ := os.UserHomeDir()
// file, err := os.Create(userHomeDir + "/Desktop/s1." + format)
// if err != nil {
// t.Fatal(err)
// }
// defer func() { _ = file.Close() }()
// switch format {
// case "png":
// err = png.Encode(file, img)
// case "jpeg":
// err = jpeg.Encode(file, img, nil)
// }
// if err != nil {
// t.Fatal(err)
// }
// t.Log(file.Name())
// save file
screenshot, err = driver.ScreenShot(option.WithScreenShotFileName("123"))
if err != nil {
t.Fatal(err)
}
_ = screenshot
path, err := saveScreenShot(screenshot, "1234")
if err != nil {
t.Fatal(err)
}
t.Logf("save screenshot to %s", path)
}
func Test_remoteWD_Source(t *testing.T) {
setup(t)
driver := setupWDADriverExt(t)
var source string
var err error
// source, err = driver.Source()
// if err != nil {
// t.Fatal(err)
// }
source, err = driver.Source()
if err != nil {
t.Fatal(err)
}
// source, err = driver.Source(NewSourceOption().WithFormatAsJson())
// if err != nil {
// t.Fatal(err)
// }
source, err = driver.Source(option.WithFormat(option.SourceFormatJSON))
if err != nil {
t.Fatal(err)
}
// source, err = driver.Source(NewSourceOption().WithFormatAsDescription())
// if err != nil {
// t.Fatal(err)
// }
source, err = driver.Source(option.WithFormat(option.SourceFormatDescription))
if err != nil {
t.Fatal(err)
}
// source, err = driver.Source(NewSourceOption().WithFormatAsXml().WithExcludedAttributes([]string{"label", "type", "index"}))
// if err != nil {
// t.Fatal(err)
// }
_ = source
fmt.Println(source)
source, err = driver.Source(
option.WithFormat(option.SourceFormatXML),
option.WithExcludedAttributes([]string{"label", "type", "index"}))
if err != nil {
t.Fatal(err)
}
t.Logf("source: %s", source)
}
func TestGetForegroundApp(t *testing.T) {
setup(t)
driver := setupWDADriverExt(t)
app, err := driver.ForegroundInfo()
if err != nil {
t.Fatal(err)
@@ -410,18 +365,17 @@ func TestGetForegroundApp(t *testing.T) {
}
func Test_remoteWD_AccessibleSource(t *testing.T) {
setup(t)
driver := setupWDADriverExt(t)
source, err := driver.(*WDADriver).AccessibleSource()
source, err := driver.IDriver.(*WDADriver).AccessibleSource()
if err != nil {
t.Fatal(err)
}
_ = source
fmt.Println(source)
}
func TestRecord(t *testing.T) {
setup(t)
driver := setupWDADriverExt(t)
path, err := driver.ScreenRecord(5 * time.Second)
if err != nil {
t.Fatal(err)
@@ -429,11 +383,11 @@ func TestRecord(t *testing.T) {
println(path)
}
// func Test_Backspace(t *testing.T) {
// setup(t)
func Test_Backspace(t *testing.T) {
driver := setupWDADriverExt(t)
// err := driver.Backspace(3)
// if err != nil {
// t.Fatal(err)
// }
// }
err := driver.Backspace(3)
if err != nil {
t.Fatal(err)
}
}

View File

@@ -13,9 +13,9 @@ type ActionOptions struct {
// control related
MaxRetryTimes int `json:"max_retry_times,omitempty" yaml:"max_retry_times,omitempty"` // max retry times
Interval float64 `json:"interval,omitempty" yaml:"interval,omitempty"` // interval between retries in seconds
Duration float64 `json:"duration,omitempty" yaml:"duration,omitempty"` // used to set duration of ios swipe action
PressDuration float64 `json:"press_duration,omitempty" yaml:"press_duration,omitempty"` // used to set duration of ios swipe action
Steps int `json:"steps,omitempty" yaml:"steps,omitempty"` // used to set steps of android swipe action
Duration float64 `json:"duration,omitempty" yaml:"duration,omitempty"` // used to set duration in seconds
PressDuration float64 `json:"press_duration,omitempty" yaml:"press_duration,omitempty"` // used to set press duration in seconds
Steps int `json:"steps,omitempty" yaml:"steps,omitempty"` // used to set steps of action
Direction interface{} `json:"direction,omitempty" yaml:"direction,omitempty"` // used by swipe to tap text or app
Timeout int `json:"timeout,omitempty" yaml:"timeout,omitempty"` // TODO: wait timeout in seconds for mobile action
Frequency int `json:"frequency,omitempty" yaml:"frequency,omitempty"`
@@ -142,7 +142,8 @@ func (o *ActionOptions) GetRandomOffset() float64 {
return float64(builtin.GetRandomNumber(minOffset, maxOffset)) + rand.Float64()
}
func (o *ActionOptions) UpdateData(data map[string]interface{}) {
func MergeOptions(data map[string]interface{}, opts ...ActionOption) {
o := NewActionOptions(opts...)
if o.Identifier != "" {
data["log"] = map[string]interface{}{
"enable": true,
@@ -164,6 +165,10 @@ func (o *ActionOptions) UpdateData(data map[string]interface{}) {
data["duration"] = 0 // default duration
}
if o.PressDuration > 0 {
data["pressDuration"] = o.PressDuration
}
if o.Frequency > 0 {
data["frequency"] = o.Frequency
}

View File

@@ -2,53 +2,74 @@ package option
import "strings"
// SourceOption Configure the format or attribute of the Source
type SourceOption map[string]interface{}
func NewSourceOption() SourceOption {
return make(SourceOption)
}
// WithFormatAsJson Application elements tree in form of json string
func (opt SourceOption) WithFormatAsJson() SourceOption {
opt["format"] = "json"
return opt
}
func (opt SourceOption) WithProcessName(processName string) SourceOption {
opt["processName"] = processName
return opt
}
// WithFormatAsXml Application elements tree in form of xml string
func (opt SourceOption) WithFormatAsXml() SourceOption {
opt["format"] = "xml"
return opt
}
// WithFormatAsDescription Application elements tree in form of internal XCTest debugDescription string
func (opt SourceOption) WithFormatAsDescription() SourceOption {
opt["format"] = "description"
return opt
}
// WithScope Allows to provide XML scope.
//
// only `xml` is supported.
func (opt SourceOption) WithScope(scope string) SourceOption {
if vFormat, ok := opt["format"]; ok && vFormat != "xml" {
return opt
func NewSourceOptions(opts ...SourceOption) *SourceOptions {
options := &SourceOptions{}
for _, option := range opts {
option(options)
}
return options
}
type SourceOptions struct {
Format SourceFormat `json:"format,omitempty"`
ProcessName string `json:"processName,omitempty"`
Scope string `json:"scope,omitempty"`
ExcludedAttributes string `json:"excluded_attributes,omitempty"`
}
func (o *SourceOptions) Query() string {
query := []string{}
if o.Format != "" {
query = append(query, "format="+string(o.Format))
}
if o.ProcessName != "" {
query = append(query, "processName="+o.ProcessName)
}
if o.Scope != "" {
query = append(query, "scope="+o.Scope)
}
if o.ExcludedAttributes != "" {
query = append(query, "excluded_attributes="+o.ExcludedAttributes)
}
return strings.Join(query, "&")
}
type SourceOption func(o *SourceOptions)
type SourceFormat string
const (
SourceFormatJSON SourceFormat = "json"
SourceFormatXML SourceFormat = "xml"
SourceFormatDescription SourceFormat = "description"
)
// WithFormat specify Application elements tree format
// `json` or `xml` or `description`
func WithFormat(format SourceFormat) SourceOption {
return func(o *SourceOptions) {
o.Format = format
}
}
func WithProcessName(name string) SourceOption {
return func(o *SourceOptions) {
o.ProcessName = name
}
}
// WithSourceScope Allows to provide XML scope.
// only `xml` is supported.
func WithSourceScope(scope string) SourceOption {
return func(o *SourceOptions) {
o.Scope = scope
}
opt["scope"] = scope
return opt
}
// WithExcludedAttributes Excludes the given attribute names.
// only `xml` is supported.
func (opt SourceOption) WithExcludedAttributes(attributes []string) SourceOption {
if vFormat, ok := opt["format"]; ok && vFormat != "xml" {
return opt
func WithExcludedAttributes(attributes []string) SourceOption {
return func(o *SourceOptions) {
o.ExcludedAttributes = strings.Join(attributes, ",")
}
opt["excluded_attributes"] = strings.Join(attributes, ",")
return opt
}