diff --git a/device.go b/device.go index 201a5c1..f15103b 100644 --- a/device.go +++ b/device.go @@ -1,10 +1,31 @@ +/* + * sonic-gidevice Connect to your iOS Devices. + * Copyright (C) 2022 SonicCloudOrg + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ package giDevice import ( "bytes" "context" + "encoding/binary" "errors" "fmt" + "io" + "log" + "net" "os" "path" "strings" @@ -28,7 +49,40 @@ func newDevice(client *libimobiledevice.UsbmuxClient, properties DevicePropertie } } +func NewRemoteConnect(ip string, port int) *device { + client, err := libimobiledevice.NewUsbmuxClient(fmt.Sprintf("%s:%d", ip, port)) + if err != nil { + log.Panic(err) + } + + clientConnectInit(client.InnerConn()) + + devicePropertiesPacket, err := client.ReceivePacket() + if err != nil { + log.Panic(err) + } + var properties = &DeviceProperties{} + err = devicePropertiesPacket.Unmarshal(properties) + if err != nil { + log.Panic(err) + } + buffer := new(bytes.Buffer) + data, err1 := client.InnerConn().Read(4) + if err1 != nil { + log.Panic(err1) + } + buffer.Write(data) + var remoteLockdownPort uint32 + if err = binary.Read(buffer, binary.LittleEndian, &remoteLockdownPort); err != nil { + log.Panic(err) + } + dev := newDevice(client, *properties) + dev.remoteAddr = fmt.Sprintf("%s:%d", ip, remoteLockdownPort) + return dev +} + type device struct { + remoteAddr string umClient *libimobiledevice.UsbmuxClient lockdownClient *libimobiledevice.LockdownClient @@ -56,11 +110,16 @@ func (d *device) Properties() DeviceProperties { } func (d *device) NewConnect(port int, timeout ...time.Duration) (InnerConn, error) { - newClient, err := libimobiledevice.NewUsbmuxClient(timeout...) + + newClient, err := libimobiledevice.NewUsbmuxClient(d.remoteAddr, timeout...) if err != nil { return nil, err } + if d.remoteAddr != "" { + clientConnectInit(newClient.InnerConn()) + } + var pkt libimobiledevice.Packet if pkt, err = newClient.NewPlistPacket( newClient.NewConnectRequest(d.properties.DeviceID, port), @@ -381,6 +440,105 @@ func (d *device) testmanagerdService() (testmanagerd Testmanagerd, err error) { return } +func (d *device) Share(port int) error { + + ln, err := net.Listen("tcp", fmt.Sprintf(":%d", port)) + if err != nil { + return err + } + address, err := net.ResolveTCPAddr("tcp", fmt.Sprintf("%s:0", "0.0.0.0")) + if err != nil { + return err + } + + lockdownLn, err := net.ListenTCP("tcp", address) + if err != nil { + return err + } + go func() { + err = d.shareBaseTcpServer(ln, lockdownLn.Addr().(*net.TCPAddr).Port) + if err != nil { + log.Panic(err) + } + }() + + return d.shareServer(lockdownLn) +} + +func (d *device) shareBaseTcpServer(ln net.Listener, serverPort int) error { + defer ln.Close() + for { + conn, err := ln.Accept() + if err != nil { + return err + } + remoteAddress := conn.RemoteAddr().String() + if debugFlag { + log.Printf("Incomming base request from: %v\n", remoteAddress) + } + + var remoteUsb = libimobiledevice.NewRemoteUsbmuxConn(conn) + localUsb, err := libimobiledevice.NewUsbmuxClient("", 0) + if err != nil { + log.Panic(err) + } + + if !serverCheckInit(remoteUsb.InnerConn()) { + continue + } + + propertiesPacket, err := remoteUsb.NewPlistPacket(d.properties) + if err != nil { + log.Panic(err) + } + err = remoteUsb.SendPacket(propertiesPacket) + if err != nil { + log.Panic(err) + } + + buf := new(bytes.Buffer) + b := make([]byte, 4) + binary.LittleEndian.PutUint32(b, uint32(serverPort)) + buf.Write(b) + _, err = conn.Write(buf.Bytes()) + if err != nil { + log.Panic(err) + } + forwardingData(conn, localUsb.RawConn()) + } +} + +func (d *device) shareServer(ln net.Listener) error { + defer ln.Close() + for { + conn, err := ln.Accept() + if err != nil { + return err + } + remoteAddress := conn.RemoteAddr().String() + + var remoteUsb = libimobiledevice.NewRemoteUsbmuxConn(conn) + + if !serverCheckInit(remoteUsb.InnerConn()) { + remoteUsb.Close() + continue + } + + if debugFlag { + log.Printf("Incomming server request from: %v\n", remoteAddress) + } + + newClient, err := libimobiledevice.NewUsbmuxClient("", 0) + if err != nil { + log.Panic(err) + } + + rConn := newClient.RawConn() + rConn.SetDeadline(time.Time{}) + forwardingData(conn, rConn) + } +} + func (d *device) AfcService() (afc Afc, err error) { if d.afc != nil { return d.afc, nil @@ -887,7 +1045,6 @@ func (d *device) XCTest(bundleID string, opts ...XCTestOption) (out <-chan strin return _out, cancelFunc, err } - // see https://github.com/SonicCloudOrg/sonic-gidevice/issues/31 // if err = d.instruments.startObserving(pid); err != nil { // return _out, cancelFunc, err // } @@ -972,3 +1129,84 @@ func (d *device) _uploadXCTestConfiguration(bundleID string, sessionId uuid.UUID return } + +func serverCheckInit(conn InnerConn) bool { + if !checkRecvMagic(conn, true) { + conn.Close() + return false + } + + buf := new(bytes.Buffer) + b := make([]byte, 4) + binary.LittleEndian.PutUint32(b, uint32(5)) + buf.Write(b) + buf.Write(checkMagic[6:]) + + err := conn.Write(buf.Bytes()) + if err != nil { + log.Panic(err) + } + return true +} + +func clientConnectInit(conn InnerConn) { + buf := new(bytes.Buffer) + b := make([]byte, 4) + + binary.LittleEndian.PutUint32(b, uint32(6)) + buf.Write(b) + buf.Write(checkMagic[:6]) + + err := conn.Write(buf.Bytes()) + if err != nil { + log.Panic(err) + } + if !checkRecvMagic(conn, false) { + conn.Close() + log.Panic("connection failed non remote sib connection") + } +} + +func forwardingData(lConn, rConn net.Conn) { + go func(lConn, rConn net.Conn) { + if _, err := io.Copy(lConn, rConn); err != nil { + lConn.Close() + rConn.Close() + if debugFlag { + log.Println(err) + } + return + } + }(lConn, rConn) + go func(lConn, rConn net.Conn) { + if _, err := io.Copy(rConn, lConn); err != nil { + lConn.Close() + rConn.Close() + if debugFlag { + log.Println(err) + } + return + } + }(lConn, rConn) +} + +// the connection is of the specified type +var checkMagic = [11]byte{0x61, 0x4F, 0x47, 0x32, 0x77, 0x6F, 0x53, 0x45, 0x45, 0x73, 0x2F} + +func checkRecvMagic(conn InnerConn, isPrefix bool) bool { + data, err := conn.Read(4) + if err != nil { + log.Panic(err) + } + data, err = conn.Read(int(binary.LittleEndian.Uint32(data))) + if err != nil { + log.Panic(err) + } + var subMagic []byte + if isPrefix { + subMagic = checkMagic[:6] + } else { + subMagic = checkMagic[6:] + } + return bytes.Equal(subMagic, data) +} diff --git a/device_test.go b/device_test.go index 00de1fc..c403550 100644 --- a/device_test.go +++ b/device_test.go @@ -2,6 +2,7 @@ package giDevice import ( "fmt" + "log" "os" "os/signal" "testing" @@ -172,3 +173,82 @@ func Test_device_InstallationProxyBrowse(t *testing.T) { t.Logf("%#v", l) } } + +func Test_device_share(t *testing.T) { + setupDevice(t) + SetDebug(true, true) + err := dev.Share(9123) + if err != nil { + log.Panic(err) + } +} + +func Test_device_connect(t *testing.T) { + SetDebug(true, true) + remoteDev := NewRemoteConnect("127.0.0.1", 9123) + data, err := remoteDev.GetValue("", "ProductVersion") + if err != nil { + log.Panic(err) + } + log.Println(data) +} + +func TestMockPerfdUser1(t *testing.T) { + //SetDebug(true, true) + remoteDev := NewRemoteConnect("127.0.0.1", 9123) + data, err := remoteDev.PerfStart( + WithPerfSystemCPU(true), + WithPerfSystemMem(true), + WithPerfSystemDisk(true), + WithPerfSystemNetwork(true), + WithPerfNetwork(true), + WithPerfFPS(true), + WithPerfGPU(true), + //WithPerfProcessAttributes("cpuUsage", "memAnon"), + //WithPerfBundleID("com.apple.mobilesafari"), + ) + if err != nil { + t.Fatal(err) + } + + timer := time.NewTimer(time.Duration(time.Second * 20)) + for { + select { + case <-timer.C: + dev.PerfStop() + return + case d := <-data: + fmt.Println(string(d)) + } + } +} + +func TestMockPerfdUser2(t *testing.T) { + //SetDebug(true, true) + remoteDev := NewRemoteConnect("127.0.0.1", 9123) + data, err := remoteDev.PerfStart( + WithPerfSystemCPU(true), + WithPerfSystemMem(true), + WithPerfSystemDisk(true), + WithPerfSystemNetwork(true), + WithPerfNetwork(true), + WithPerfFPS(true), + WithPerfGPU(true), + //WithPerfProcessAttributes("cpuUsage", "memAnon"), + //WithPerfBundleID("com.apple.mobilesafari"), + ) + if err != nil { + t.Fatal(err) + } + + timer := time.NewTimer(time.Duration(time.Second * 10)) + for { + select { + case <-timer.C: + dev.PerfStop() + return + case d := <-data: + fmt.Println(string(d)) + } + } +} diff --git a/idevice.go b/idevice.go index 7f83a09..3da5147 100644 --- a/idevice.go +++ b/idevice.go @@ -81,6 +81,8 @@ type Device interface { PerfStop() WebInspectorService() (webInspector WebInspector, err error) + + Share(port int) error } type DeviceProperties = libimobiledevice.DeviceProperties diff --git a/pkg/libimobiledevice/client_dtxmessage.go b/pkg/libimobiledevice/client_dtxmessage.go index 7fed570..7c03e35 100644 --- a/pkg/libimobiledevice/client_dtxmessage.go +++ b/pkg/libimobiledevice/client_dtxmessage.go @@ -179,7 +179,6 @@ func (c *dtxMessageClient) ReceiveDTXMessage() (result *DTXMessageResult, err er var aux, obj []byte - // see https://github.com/SonicCloudOrg/sonic-gidevice/issues/28 if r, l := payloadSize+payload.AuxiliaryLength, len(rawPayload); int(r) <= l { aux = rawPayload[payloadSize:r] } else { diff --git a/pkg/libimobiledevice/usbmux.go b/pkg/libimobiledevice/usbmux.go index 06b32a1..2d7743c 100644 --- a/pkg/libimobiledevice/usbmux.go +++ b/pkg/libimobiledevice/usbmux.go @@ -1,3 +1,20 @@ +/* + * sonic-gidevice Connect to your iOS Devices. + * Copyright (C) 2022 SonicCloudOrg + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ package libimobiledevice import ( @@ -15,7 +32,7 @@ import ( var DefaultDeadlineTimeout = 30 * time.Second const ( - BundleID = "electricbubble.libimobiledevice" + BundleID = "org.cloud.sonic.gidevice" ProgramName = "libimobiledevice" ClientVersion = "libimobiledevice-beta" LibUSBMuxVersion = 3 @@ -112,13 +129,13 @@ type DeviceProperties struct { NetworkAddress []byte `plist:"NetworkAddress"` } -func NewUsbmuxClient(timeout ...time.Duration) (c *UsbmuxClient, err error) { +func NewUsbmuxClient(addr string, timeout ...time.Duration) (c *UsbmuxClient, err error) { if len(timeout) == 0 { timeout = []time.Duration{DefaultDeadlineTimeout} } c = &UsbmuxClient{version: ProtoVersionPlist} var conn net.Conn - if conn, err = rawDial(timeout[0]); err != nil { + if conn, err = rawDial(addr, timeout[0]); err != nil { return nil, fmt.Errorf("usbmux connect: %w", err) } @@ -126,6 +143,12 @@ func NewUsbmuxClient(timeout ...time.Duration) (c *UsbmuxClient, err error) { return } +func NewRemoteUsbmuxConn(conn net.Conn) (c *UsbmuxClient) { + c = &UsbmuxClient{version: ProtoVersionPlist} + c.innerConn = newInnerConn(conn, 0) + return +} + type UsbmuxClient struct { innerConn InnerConn version ProtoVersion @@ -259,11 +282,13 @@ func (c *UsbmuxClient) InnerConn() InnerConn { return c.innerConn } -func rawDial(timeout time.Duration) (net.Conn, error) { +func rawDial(addr string, timeout time.Duration) (net.Conn, error) { dialer := net.Dialer{ Timeout: timeout, } - + if addr != "" { + return dialer.Dial("tcp", addr) + } var network, address string switch runtime.GOOS { case "darwin", "android", "linux": diff --git a/pkg/libimobiledevice/webinspector.go b/pkg/libimobiledevice/webinspector.go index 1f7f521..6d6de5d 100644 --- a/pkg/libimobiledevice/webinspector.go +++ b/pkg/libimobiledevice/webinspector.go @@ -1,3 +1,20 @@ +/* + * sonic-gidevice Connect to your iOS Devices. + * Copyright (C) 2022 SonicCloudOrg + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ package libimobiledevice import ( diff --git a/usbmux.go b/usbmux.go index c6a986f..6050536 100644 --- a/usbmux.go +++ b/usbmux.go @@ -9,7 +9,7 @@ import ( var _ Usbmux = (*usbmux)(nil) func NewUsbmux() (Usbmux, error) { - umClient, err := libimobiledevice.NewUsbmuxClient() + umClient, err := libimobiledevice.NewUsbmuxClient("") if err != nil { return nil, err } @@ -99,7 +99,7 @@ func (um *usbmux) Listen(devNotifier chan Device) (context.CancelFunc, error) { if baseDev.MessageType != libimobiledevice.MessageTypeDeviceAdd { baseDev.Properties.DeviceID = baseDev.DeviceID } - client, err := libimobiledevice.NewUsbmuxClient() + client, err := libimobiledevice.NewUsbmuxClient("") if err != nil { continue }