Compare commits
6 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
5f90ebb32b | ||
|
|
54a62bfdd0 | ||
|
|
3799a8a535 | ||
|
|
e466aeac66 | ||
|
|
ffe40a0fac | ||
|
|
c5d206b72c |
5
.gitignore
vendored
5
.gitignore
vendored
@@ -1,5 +1,6 @@
|
||||
demo
|
||||
demo.app
|
||||
ios-deploy
|
||||
|
||||
/.DS_Store
|
||||
ios-deploy.dSYM
|
||||
/.DS_Store
|
||||
*~
|
||||
|
||||
14
README.md
14
README.md
@@ -10,7 +10,7 @@ Install and debug iPhone apps without using Xcode. Designed to work on unjailbro
|
||||
|
||||
## Usage
|
||||
|
||||
./ios-deploy [OPTION]...
|
||||
Usage: ./ios-deploy [OPTION]...
|
||||
-d, --debug launch the app in GDB after installation
|
||||
-i, --id <device_id> the id of the device to connect to
|
||||
-c, --detect only detect if the device is connected
|
||||
@@ -23,10 +23,14 @@ Install and debug iPhone apps without using Xcode. Designed to work on unjailbro
|
||||
-n, --nostart do not start the app when debugging
|
||||
-I, --noninteractive start in non interactive mode (quit when app crashes or exits)
|
||||
-v, --verbose enable verbose output
|
||||
-m, --noinstall directly start debugging without app install (-d not required)
|
||||
-p, --port <number> port used for device, default: 12345
|
||||
-r, --uninstall uninstall the app before install (do not use with -m; app cache and data are cleared)
|
||||
-V, --version print the executable version
|
||||
-m, --noinstall directly start debugging without app install (-d not required)
|
||||
-p, --port <number> port used for device, default: 12345
|
||||
-r, --uninstall uninstall the app before install (do not use with -m; app cache and data are cleared)
|
||||
-1, --bundle_id <bundle id> specify bundle id for list and upload
|
||||
-l, --list list files
|
||||
-o, --upload <file> upload file
|
||||
-2, --to <target pathname> use together with upload file. specify target for upload
|
||||
-V, --version print the executable version
|
||||
|
||||
## Demo
|
||||
|
||||
|
||||
439
ios-deploy.c
439
ios-deploy.c
@@ -16,7 +16,7 @@
|
||||
#include <netinet/tcp.h>
|
||||
#include "MobileDevice.h"
|
||||
|
||||
#define APP_VERSION "1.0.8"
|
||||
#define APP_VERSION "1.0.9"
|
||||
#define PREP_CMDS_PATH "/tmp/fruitstrap-lldb-prep-cmds-"
|
||||
#define LLDB_SHELL "lldb -s " PREP_CMDS_PATH
|
||||
|
||||
@@ -57,6 +57,7 @@ const char* lldb_prep_noninteractive_cmds = "\
|
||||
#define LLDB_FRUITSTRAP_MODULE CFSTR("\
|
||||
import lldb\n\
|
||||
import sys\n\
|
||||
import shlex\n\
|
||||
\n\
|
||||
def connect_command(debugger, command, result, internal_dict):\n\
|
||||
# These two are passed in by the script which loads us\n\
|
||||
@@ -86,7 +87,7 @@ def run_command(debugger, command, result, internal_dict):\n\
|
||||
device_app = internal_dict['fruitstrap_device_app']\n\
|
||||
error = lldb.SBError()\n\
|
||||
lldb.target.modules[0].SetPlatformFileSpec(lldb.SBFileSpec(device_app))\n\
|
||||
lldb.target.Launch(lldb.SBLaunchInfo(['{args}']), error)\n\
|
||||
lldb.target.Launch(lldb.SBLaunchInfo(shlex.split('{args}')), error)\n\
|
||||
print str(error)\n\
|
||||
\n\
|
||||
def autoexit_command(debugger, command, result, internal_dict):\n\
|
||||
@@ -120,12 +121,19 @@ def autoexit_command(debugger, command, result, internal_dict):\n\
|
||||
")
|
||||
|
||||
typedef struct am_device * AMDeviceRef;
|
||||
mach_error_t AMDeviceSecureStartService(struct am_device *device, CFStringRef service_name, unsigned int *unknown, service_conn_t *handle);
|
||||
int AMDeviceSecureTransferPath(int zero, AMDeviceRef device, CFURLRef url, CFDictionaryRef options, void *callback, int cbarg);
|
||||
int AMDeviceSecureInstallApplication(int zero, AMDeviceRef device, CFURLRef url, CFDictionaryRef options, void *callback, int cbarg);
|
||||
int AMDeviceMountImage(AMDeviceRef device, CFStringRef image, CFDictionaryRef options, void *callback, int cbarg);
|
||||
mach_error_t AMDeviceLookupApplications(AMDeviceRef device, CFDictionaryRef options, CFDictionaryRef *result);
|
||||
int AMDeviceGetInterfaceType(struct am_device *device);
|
||||
|
||||
bool found_device = false, debug = false, verbose = false, unbuffered = false, nostart = false, detect_only = false, install = true, uninstall = false;
|
||||
bool command_only = false;
|
||||
char *command = NULL;
|
||||
char *target_filename = NULL;
|
||||
char *upload_pathname = NULL;
|
||||
char *bundle_id = NULL;
|
||||
bool interactive = true;
|
||||
char *app_path = NULL;
|
||||
char *device_id = NULL;
|
||||
@@ -139,6 +147,7 @@ pid_t parent = 0;
|
||||
pid_t child = 0;
|
||||
// Signal sent from child to parent process when LLDB finishes.
|
||||
const int SIGLLDB = SIGUSR1;
|
||||
AMDeviceRef best_device_match = NULL;
|
||||
|
||||
// Error codes we report on different failures, so scripts can distinguish between user app exit
|
||||
// codes and our exit codes. For non app errors we use codes in reserved 128-255 range.
|
||||
@@ -262,6 +271,125 @@ CFStringRef copy_xcode_path_for(CFStringRef subPath, CFStringRef search) {
|
||||
}
|
||||
}
|
||||
|
||||
// Please ensure that device is connected or the name will be unknown
|
||||
const CFStringRef get_device_hardware_name(const AMDeviceRef device) {
|
||||
CFStringRef model = AMDeviceCopyValue(device, 0, CFSTR("HardwareModel"));
|
||||
const char *hwmodel = CFStringGetCStringPtr(model, CFStringGetSystemEncoding());
|
||||
|
||||
if (hwmodel && !strcmp("M68AP", hwmodel))
|
||||
return CFSTR("iPhone");
|
||||
if (hwmodel && !strcmp("N45AP", hwmodel))
|
||||
return CFSTR("iPod touch");
|
||||
if (hwmodel && !strcmp("N82AP", hwmodel))
|
||||
return CFSTR("iPhone 3G");
|
||||
if (hwmodel && !strcmp("N72AP", hwmodel))
|
||||
return CFSTR("iPod touch 2G");
|
||||
if (hwmodel && !strcmp("N88AP", hwmodel))
|
||||
return CFSTR("iPhone 3GS");
|
||||
if (hwmodel && !strcmp("N18AP", hwmodel))
|
||||
return CFSTR("iPod touch 3G");
|
||||
if (hwmodel && !strcmp("K48AP", hwmodel))
|
||||
return CFSTR("iPad");
|
||||
if (hwmodel && !strcmp("N90AP", hwmodel))
|
||||
return CFSTR("iPhone 4 (GSM)");
|
||||
if (hwmodel && !strcmp("N81AP", hwmodel))
|
||||
return CFSTR("iPod touch 4G");
|
||||
if (hwmodel && !strcmp("K66AP", hwmodel))
|
||||
return CFSTR("Apple TV 2G");
|
||||
if (hwmodel && !strcmp("N92AP", hwmodel))
|
||||
return CFSTR("iPhone 4 (CDMA)");
|
||||
if (hwmodel && !strcmp("N90BAP", hwmodel))
|
||||
return CFSTR("iPhone 4 (GSM, revision A)");
|
||||
if (hwmodel && !strcmp("K93AP", hwmodel))
|
||||
return CFSTR("iPad 2");
|
||||
if (hwmodel && !strcmp("K94AP", hwmodel))
|
||||
return CFSTR("iPad 2 (GSM)");
|
||||
if (hwmodel && !strcmp("K95AP", hwmodel))
|
||||
return CFSTR("iPad 2 (CDMA)");
|
||||
if (hwmodel && !strcmp("K93AAP", hwmodel))
|
||||
return CFSTR("iPad 2 (Wi-Fi, revision A)");
|
||||
if (hwmodel && !strcmp("P105AP", hwmodel))
|
||||
return CFSTR("iPad mini");
|
||||
if (hwmodel && !strcmp("P106AP", hwmodel))
|
||||
return CFSTR("iPad mini (GSM)");
|
||||
if (hwmodel && !strcmp("P107AP", hwmodel))
|
||||
return CFSTR("iPad mini (CDMA)");
|
||||
if (hwmodel && !strcmp("N94AP", hwmodel))
|
||||
return CFSTR("iPhone 4S");
|
||||
if (hwmodel && !strcmp("N41AP", hwmodel))
|
||||
return CFSTR("iPhone 5 (GSM)");
|
||||
if (hwmodel && !strcmp("N42AP", hwmodel))
|
||||
return CFSTR("iPhone 5 (Global/CDMA)");
|
||||
if (hwmodel && !strcmp("N48AP", hwmodel))
|
||||
return CFSTR("iPhone 5c (GSM)");
|
||||
if (hwmodel && !strcmp("N49AP", hwmodel))
|
||||
return CFSTR("iPhone 5c (Global/CDMA)");
|
||||
if (hwmodel && !strcmp("N51AP", hwmodel))
|
||||
return CFSTR("iPhone 5s (GSM)");
|
||||
if (hwmodel && !strcmp("N53AP", hwmodel))
|
||||
return CFSTR("iPhone 5s (Global/CDMA)");
|
||||
if (hwmodel && !strcmp("J1AP", hwmodel))
|
||||
return CFSTR("iPad 3");
|
||||
if (hwmodel && !strcmp("J2AP", hwmodel))
|
||||
return CFSTR("iPad 3 (GSM)");
|
||||
if (hwmodel && !strcmp("J2AAP", hwmodel))
|
||||
return CFSTR("iPad 3 (CDMA)");
|
||||
if (hwmodel && !strcmp("P101AP", hwmodel))
|
||||
return CFSTR("iPad 4");
|
||||
if (hwmodel && !strcmp("P102AP", hwmodel))
|
||||
return CFSTR("iPad 4 (GSM)");
|
||||
if (hwmodel && !strcmp("P103AP", hwmodel))
|
||||
return CFSTR("iPad 4 (CDMA)");
|
||||
if (hwmodel && !strcmp("N78AP", hwmodel))
|
||||
return CFSTR("iPod touch 5G");
|
||||
if (hwmodel && !strcmp("J33AP", hwmodel))
|
||||
return CFSTR("Apple TV 3G");
|
||||
if (hwmodel && !strcmp("J33IAP", hwmodel))
|
||||
return CFSTR("Apple TV 3.1G");
|
||||
|
||||
return CFSTR("Unknown Device");
|
||||
}
|
||||
|
||||
CFStringRef get_device_full_name(const AMDeviceRef device) {
|
||||
CFStringRef full_name = NULL,
|
||||
device_udid = AMDeviceCopyDeviceIdentifier(device),
|
||||
device_name = NULL,
|
||||
model_name = NULL;
|
||||
|
||||
AMDeviceConnect(device);
|
||||
|
||||
device_name = AMDeviceCopyValue(device, 0, CFSTR("DeviceName")),
|
||||
model_name = get_device_hardware_name(device);
|
||||
|
||||
if(device_name != NULL && model_name != NULL)
|
||||
full_name = CFStringCreateWithFormat(NULL, NULL, CFSTR("%@ '%@' (%@)"), model_name, device_name, device_udid);
|
||||
else
|
||||
full_name = CFStringCreateWithFormat(NULL, NULL, CFSTR("(%@)"), device_udid);
|
||||
|
||||
AMDeviceDisconnect(device);
|
||||
|
||||
if(device_udid != NULL)
|
||||
CFRelease(device_udid);
|
||||
if(device_name != NULL)
|
||||
CFRelease(device_name);
|
||||
if(model_name != NULL)
|
||||
CFRelease(model_name);
|
||||
|
||||
return full_name;
|
||||
}
|
||||
|
||||
CFStringRef get_device_interface_name(const AMDeviceRef device) {
|
||||
// AMDeviceGetInterfaceType(device) 0=Unknown, 1 = Direct/USB, 2 = Indirect/WIFI
|
||||
switch(AMDeviceGetInterfaceType(device)) {
|
||||
case 1:
|
||||
return CFSTR("USB");
|
||||
case 2:
|
||||
return CFSTR("WIFI");
|
||||
default:
|
||||
return CFSTR("Unknown Connection");
|
||||
}
|
||||
}
|
||||
|
||||
CFMutableArrayRef get_device_product_version_parts(AMDeviceRef device) {
|
||||
CFStringRef version = AMDeviceCopyValue(device, 0, CFSTR("ProductVersion"));
|
||||
CFArrayRef parts = CFStringCreateArrayBySeparatingStrings(NULL, version, CFSTR("."));
|
||||
@@ -500,15 +628,8 @@ void write_lldb_prep_cmds(AMDeviceRef device, CFURLRef disk_app_url) {
|
||||
if (args) {
|
||||
CFStringRef cf_args = CFStringCreateWithCString(NULL, args, kCFStringEncodingASCII);
|
||||
CFStringFindAndReplace(cmds, CFSTR("{args}"), cf_args, range, 0);
|
||||
|
||||
//format the arguments 'arg1 arg2 ....' to an argument list ['arg1', 'arg2', ...]
|
||||
CFMutableStringRef argsListLLDB = CFStringCreateMutableCopy(NULL, 0, cf_args);
|
||||
rangeLLDB.length = CFStringGetLength(argsListLLDB);
|
||||
CFStringFindAndReplace(argsListLLDB, CFSTR(" "), CFSTR(" "), rangeLLDB, 0);//remove multiple spaces
|
||||
rangeLLDB.length = CFStringGetLength(argsListLLDB);
|
||||
CFStringFindAndReplace(argsListLLDB, CFSTR(" "), CFSTR("', '"), rangeLLDB, 0);
|
||||
rangeLLDB.length = CFStringGetLength(pmodule);
|
||||
CFStringFindAndReplace(pmodule, CFSTR("{args}"), argsListLLDB, rangeLLDB, 0);
|
||||
CFStringFindAndReplace(pmodule, CFSTR("{args}"), cf_args, rangeLLDB, 0);
|
||||
|
||||
CFRelease(cf_args);
|
||||
} else {
|
||||
@@ -749,6 +870,9 @@ void setup_dummy_pipe_on_stdin(int pfd[2]) {
|
||||
}
|
||||
|
||||
void launch_debugger(AMDeviceRef device, CFURLRef url) {
|
||||
CFStringRef device_full_name = get_device_full_name(device),
|
||||
device_interface_name = get_device_interface_name(device);
|
||||
|
||||
AMDeviceConnect(device);
|
||||
assert(AMDeviceIsPaired(device));
|
||||
assert(AMDeviceValidatePairing(device) == 0);
|
||||
@@ -756,6 +880,14 @@ void launch_debugger(AMDeviceRef device, CFURLRef url) {
|
||||
|
||||
printf("------ Debug phase ------\n");
|
||||
|
||||
if(AMDeviceGetInterfaceType(device) == 2)
|
||||
{
|
||||
printf("Cannot debug %s over %s.\n", CFStringGetCStringPtr(device_full_name, CFStringGetSystemEncoding()), CFStringGetCStringPtr(device_interface_name, CFStringGetSystemEncoding()));
|
||||
exit(0);
|
||||
}
|
||||
|
||||
printf("Starting debug of %s connected through %s...\n", CFStringGetCStringPtr(device_full_name, CFStringGetSystemEncoding()), CFStringGetCStringPtr(device_interface_name, CFStringGetSystemEncoding()));
|
||||
|
||||
mount_developer_image(device); // put debugserver on the device
|
||||
start_remote_debug_server(device); // start debugserver
|
||||
write_lldb_prep_cmds(device, url); // dump the necessary lldb commands into a file
|
||||
@@ -800,7 +932,7 @@ void launch_debugger(AMDeviceRef device, CFURLRef url) {
|
||||
perror("failed launching lldb");
|
||||
|
||||
close(pfd[0]);
|
||||
close(pfd[1]);
|
||||
close(pfd[1]);
|
||||
// Notify parent we're exiting
|
||||
kill(parent, SIGLLDB);
|
||||
// Pass lldb exit code
|
||||
@@ -849,9 +981,194 @@ CFStringRef get_bundle_id(CFURLRef app_url)
|
||||
return bundle_id;
|
||||
}
|
||||
|
||||
void read_dir(service_conn_t afcFd, afc_connection* afc_conn_p, const char* dir)
|
||||
{
|
||||
char *dir_ent;
|
||||
|
||||
afc_connection afc_conn;
|
||||
if (!afc_conn_p) {
|
||||
afc_conn_p = &afc_conn;
|
||||
AFCConnectionOpen(afcFd, 0, &afc_conn_p);
|
||||
}
|
||||
|
||||
printf("%s\n", dir);
|
||||
|
||||
afc_dictionary afc_dict;
|
||||
afc_dictionary* afc_dict_p = &afc_dict;
|
||||
AFCFileInfoOpen(afc_conn_p, dir, &afc_dict_p);
|
||||
|
||||
afc_directory afc_dir;
|
||||
afc_directory* afc_dir_p = &afc_dir;
|
||||
afc_error_t err = AFCDirectoryOpen(afc_conn_p, dir, &afc_dir_p);
|
||||
|
||||
if (err != 0)
|
||||
{
|
||||
// Couldn't open dir - was probably a file
|
||||
return;
|
||||
}
|
||||
|
||||
while(true) {
|
||||
err = AFCDirectoryRead(afc_conn_p, afc_dir_p, &dir_ent);
|
||||
|
||||
if (!dir_ent)
|
||||
break;
|
||||
|
||||
if (strcmp(dir_ent, ".") == 0 || strcmp(dir_ent, "..") == 0)
|
||||
continue;
|
||||
|
||||
char* dir_joined = malloc(strlen(dir) + strlen(dir_ent) + 2);
|
||||
strcpy(dir_joined, dir);
|
||||
if (dir_joined[strlen(dir)-1] != '/')
|
||||
strcat(dir_joined, "/");
|
||||
strcat(dir_joined, dir_ent);
|
||||
read_dir(afcFd, afc_conn_p, dir_joined);
|
||||
free(dir_joined);
|
||||
}
|
||||
|
||||
AFCDirectoryClose(afc_conn_p, afc_dir_p);
|
||||
}
|
||||
|
||||
|
||||
// Used to send files to app-specific sandbox (Documents dir)
|
||||
service_conn_t start_house_arrest_service(AMDeviceRef device) {
|
||||
AMDeviceConnect(device);
|
||||
assert(AMDeviceIsPaired(device));
|
||||
assert(AMDeviceValidatePairing(device) == 0);
|
||||
assert(AMDeviceStartSession(device) == 0);
|
||||
|
||||
service_conn_t houseFd;
|
||||
|
||||
if (bundle_id == NULL) {
|
||||
printf("Bundle id is not specified\n");
|
||||
exit(1);
|
||||
}
|
||||
|
||||
CFStringRef cf_bundle_id = CFStringCreateWithCString(NULL, bundle_id, kCFStringEncodingASCII);
|
||||
if (AMDeviceStartHouseArrestService(device, cf_bundle_id, 0, &houseFd, 0) != 0)
|
||||
{
|
||||
printf("Unable to find bundle with id: %s\n", bundle_id);
|
||||
exit(1);
|
||||
}
|
||||
|
||||
assert(AMDeviceStopSession(device) == 0);
|
||||
assert(AMDeviceDisconnect(device) == 0);
|
||||
CFRelease(cf_bundle_id);
|
||||
|
||||
return houseFd;
|
||||
}
|
||||
|
||||
char* get_filename_from_path(char* path)
|
||||
{
|
||||
char *ptr = path + strlen(path);
|
||||
while (ptr > path)
|
||||
{
|
||||
if (*ptr == '/')
|
||||
break;
|
||||
--ptr;
|
||||
}
|
||||
if (ptr+1 >= path+strlen(path))
|
||||
return NULL;
|
||||
if (ptr == path)
|
||||
return ptr;
|
||||
return ptr+1;
|
||||
}
|
||||
|
||||
void* read_file_to_memory(char * path, size_t* file_size)
|
||||
{
|
||||
struct stat buf;
|
||||
int err = stat(path, &buf);
|
||||
if (err < 0)
|
||||
{
|
||||
return NULL;
|
||||
}
|
||||
|
||||
*file_size = buf.st_size;
|
||||
FILE* fd = fopen(path, "r");
|
||||
char* content = malloc(*file_size);
|
||||
if (fread(content, *file_size, 1, fd) != 1)
|
||||
{
|
||||
fclose(fd);
|
||||
return NULL;
|
||||
}
|
||||
fclose(fd);
|
||||
return content;
|
||||
}
|
||||
|
||||
void list_files(AMDeviceRef device)
|
||||
{
|
||||
service_conn_t houseFd = start_house_arrest_service(device);
|
||||
|
||||
afc_connection afc_conn;
|
||||
afc_connection* afc_conn_p = &afc_conn;
|
||||
AFCConnectionOpen(houseFd, 0, &afc_conn_p);
|
||||
|
||||
read_dir(houseFd, afc_conn_p, "/");
|
||||
}
|
||||
|
||||
void upload_file(AMDeviceRef device) {
|
||||
service_conn_t houseFd = start_house_arrest_service(device);
|
||||
|
||||
afc_file_ref file_ref;
|
||||
|
||||
afc_connection afc_conn;
|
||||
afc_connection* afc_conn_p = &afc_conn;
|
||||
AFCConnectionOpen(houseFd, 0, &afc_conn_p);
|
||||
|
||||
// read_dir(houseFd, NULL, "/");
|
||||
|
||||
if (!target_filename)
|
||||
{
|
||||
target_filename = get_filename_from_path(upload_pathname);
|
||||
}
|
||||
|
||||
size_t file_size;
|
||||
void* file_content = read_file_to_memory(upload_pathname, &file_size);
|
||||
|
||||
if (!file_content)
|
||||
{
|
||||
printf("Could not open file: %s\n", upload_pathname);
|
||||
exit(-1);
|
||||
}
|
||||
|
||||
// Make sure the directory was created
|
||||
{
|
||||
char *dirpath = strdup(target_filename);
|
||||
char *c = dirpath, *lastSlash = dirpath;
|
||||
while(*c) {
|
||||
if(*c == '/') {
|
||||
lastSlash = c;
|
||||
}
|
||||
c++;
|
||||
}
|
||||
*lastSlash = '\0';
|
||||
assert(AFCDirectoryCreate(afc_conn_p, dirpath) == 0);
|
||||
}
|
||||
|
||||
|
||||
int ret = AFCFileRefOpen(afc_conn_p, target_filename, 3, &file_ref);
|
||||
if (ret == 0x000a) {
|
||||
printf("Cannot write to %s. Permission error.\n", target_filename);
|
||||
exit(1);
|
||||
}
|
||||
if (ret == 0x0009) {
|
||||
printf("Target %s is a directory.\n", target_filename);
|
||||
exit(1);
|
||||
}
|
||||
assert(ret == 0);
|
||||
assert(AFCFileRefWrite(afc_conn_p, file_ref, file_content, file_size) == 0);
|
||||
assert(AFCFileRefClose(afc_conn_p, file_ref) == 0);
|
||||
assert(AFCConnectionClose(afc_conn_p) == 0);
|
||||
|
||||
free(file_content);
|
||||
}
|
||||
|
||||
void handle_device(AMDeviceRef device) {
|
||||
if (found_device) return; // handle one device only
|
||||
CFStringRef found_device_id = AMDeviceCopyDeviceIdentifier(device);
|
||||
if (found_device)
|
||||
return; // handle one device only
|
||||
|
||||
CFStringRef found_device_id = AMDeviceCopyDeviceIdentifier(device),
|
||||
device_full_name = get_device_full_name(device),
|
||||
device_interface_name = get_device_interface_name(device);
|
||||
|
||||
if (device_id != NULL) {
|
||||
if(strcmp(device_id, CFStringGetCStringPtr(found_device_id, CFStringGetSystemEncoding())) == 0) {
|
||||
@@ -864,7 +1181,16 @@ void handle_device(AMDeviceRef device) {
|
||||
}
|
||||
|
||||
if (detect_only) {
|
||||
printf("[....] Found device (%s).\n", CFStringGetCStringPtr(found_device_id, CFStringGetSystemEncoding()));
|
||||
printf("[....] Found %s connected through %s.\n", CFStringGetCStringPtr(device_full_name, CFStringGetSystemEncoding()), CFStringGetCStringPtr(device_interface_name, CFStringGetSystemEncoding()));
|
||||
exit(0);
|
||||
}
|
||||
|
||||
if (command_only) {
|
||||
if (strcmp("list", command) == 0) {
|
||||
list_files(device);
|
||||
} else if (strcmp("upload", command) == 0) {
|
||||
upload_file(device);
|
||||
}
|
||||
exit(0);
|
||||
}
|
||||
|
||||
@@ -900,7 +1226,7 @@ void handle_device(AMDeviceRef device) {
|
||||
|
||||
if(install) {
|
||||
printf("------ Install phase ------\n");
|
||||
printf("[ 0%%] Found device (%s), beginning install\n", CFStringGetCStringPtr(found_device_id, CFStringGetSystemEncoding()));
|
||||
printf("[ 0%%] Found %s connected through %s, beginning install\n", CFStringGetCStringPtr(device_full_name, CFStringGetSystemEncoding()), CFStringGetCStringPtr(device_interface_name, CFStringGetSystemEncoding()));
|
||||
|
||||
AMDeviceConnect(device);
|
||||
assert(AMDeviceIsPaired(device));
|
||||
@@ -908,31 +1234,37 @@ void handle_device(AMDeviceRef device) {
|
||||
assert(AMDeviceStartSession(device) == 0);
|
||||
|
||||
|
||||
|
||||
// NOTE: the secure version doesn't seem to require us to start the AFC service
|
||||
service_conn_t afcFd;
|
||||
assert(AMDeviceStartService(device, CFSTR("com.apple.afc"), &afcFd, NULL) == 0);
|
||||
assert(AMDeviceSecureStartService(device, CFSTR("com.apple.afc"), NULL, &afcFd) == 0);
|
||||
assert(AMDeviceStopSession(device) == 0);
|
||||
assert(AMDeviceDisconnect(device) == 0);
|
||||
assert(AMDeviceTransferApplication(afcFd, path, NULL, transfer_callback, NULL) == 0);
|
||||
|
||||
close(afcFd);
|
||||
|
||||
CFStringRef keys[] = { CFSTR("PackageType") };
|
||||
CFStringRef values[] = { CFSTR("Developer") };
|
||||
CFDictionaryRef options = CFDictionaryCreate(NULL, (const void **)&keys, (const void **)&values, 1, &kCFTypeDictionaryKeyCallBacks, &kCFTypeDictionaryValueCallBacks);
|
||||
|
||||
//assert(AMDeviceTransferApplication(afcFd, path, NULL, transfer_callback, NULL) == 0);
|
||||
assert(AMDeviceSecureTransferPath(0, device, url, options, transfer_callback, 0)==0);
|
||||
|
||||
close(afcFd);
|
||||
|
||||
|
||||
|
||||
AMDeviceConnect(device);
|
||||
assert(AMDeviceIsPaired(device));
|
||||
assert(AMDeviceValidatePairing(device) == 0);
|
||||
assert(AMDeviceStartSession(device) == 0);
|
||||
|
||||
service_conn_t installFd;
|
||||
assert(AMDeviceStartService(device, CFSTR("com.apple.mobile.installation_proxy"), &installFd, NULL) == 0);
|
||||
// // NOTE: the secure version doesn't seem to require us to start the installation_proxy service
|
||||
// // Although I can't find it right now, I in some code that the first param of AMDeviceSecureInstallApplication was a "dontStartInstallProxy"
|
||||
// // implying this is done for us by iOS already
|
||||
|
||||
assert(AMDeviceStopSession(device) == 0);
|
||||
assert(AMDeviceDisconnect(device) == 0);
|
||||
//service_conn_t installFd;
|
||||
//assert(AMDeviceSecureStartService(device, CFSTR("com.apple.mobile.installation_proxy"), NULL, &installFd) == 0);
|
||||
|
||||
mach_error_t result = AMDeviceInstallApplication(installFd, path, options, install_callback, NULL);
|
||||
//mach_error_t result = AMDeviceInstallApplication(installFd, path, options, install_callback, NULL);
|
||||
mach_error_t result = AMDeviceSecureInstallApplication(0, device, url, options, install_callback, 0);
|
||||
if (result != 0)
|
||||
{
|
||||
char* error = "Unknown error.";
|
||||
@@ -942,7 +1274,10 @@ void handle_device(AMDeviceRef device) {
|
||||
exit(exitcode_error);
|
||||
}
|
||||
|
||||
close(installFd);
|
||||
// close(installFd);
|
||||
|
||||
assert(AMDeviceStopSession(device) == 0);
|
||||
assert(AMDeviceDisconnect(device) == 0);
|
||||
|
||||
CFRelease(path);
|
||||
CFRelease(options);
|
||||
@@ -950,14 +1285,21 @@ void handle_device(AMDeviceRef device) {
|
||||
printf("[100%%] Installed package %s\n", app_path);
|
||||
}
|
||||
|
||||
if (!debug) exit(0); // no debug phase
|
||||
if (!debug)
|
||||
exit(0); // no debug phase
|
||||
|
||||
launch_debugger(device, url);
|
||||
}
|
||||
|
||||
void device_callback(struct am_device_notification_callback_info *info, void *arg) {
|
||||
switch (info->msg) {
|
||||
case ADNCI_MSG_CONNECTED:
|
||||
handle_device(info->dev);
|
||||
if(device_id != NULL || !debug || AMDeviceGetInterfaceType(info->dev) != 2) {
|
||||
handle_device(info->dev);
|
||||
} else if(best_device_match == NULL) {
|
||||
best_device_match = info->dev;
|
||||
CFRetain(best_device_match);
|
||||
}
|
||||
default:
|
||||
break;
|
||||
}
|
||||
@@ -965,8 +1307,17 @@ void device_callback(struct am_device_notification_callback_info *info, void *ar
|
||||
|
||||
void timeout_callback(CFRunLoopTimerRef timer, void *info) {
|
||||
if (!found_device) {
|
||||
printf("[....] Timed out waiting for device.\n");
|
||||
exit(exitcode_error);
|
||||
if(best_device_match != NULL) {
|
||||
handle_device(best_device_match);
|
||||
|
||||
CFRelease(best_device_match);
|
||||
best_device_match = NULL;
|
||||
}
|
||||
|
||||
if(!found_device) {
|
||||
printf("[....] Timed out waiting for device.\n");
|
||||
exit(exitcode_error);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -988,6 +1339,10 @@ void usage(const char* app) {
|
||||
" -m, --noinstall directly start debugging without app install (-d not required)\n"
|
||||
" -p, --port <number> port used for device, default: 12345 \n"
|
||||
" -r, --uninstall uninstall the app before install (do not use with -m; app cache and data are cleared) \n"
|
||||
" -1, --bundle_id <bundle id> specify bundle id for list and upload\n"
|
||||
" -l, --list list files\n"
|
||||
" -o, --upload <file> upload file\n"
|
||||
" -2, --to <target pathname> use together with upload file. specify target for upload\n"
|
||||
" -V, --version print the executable version \n",
|
||||
app);
|
||||
}
|
||||
@@ -1013,11 +1368,15 @@ int main(int argc, char *argv[]) {
|
||||
{ "noinstall", no_argument, NULL, 'm' },
|
||||
{ "port", required_argument, NULL, 'p' },
|
||||
{ "uninstall", no_argument, NULL, 'r' },
|
||||
{ "list", no_argument, NULL, 'l' },
|
||||
{ "bundle_id", required_argument, NULL, '1'},
|
||||
{ "upload", required_argument, NULL, 'o'},
|
||||
{ "to", required_argument, NULL, '2'},
|
||||
{ NULL, 0, NULL, 0 },
|
||||
};
|
||||
char ch;
|
||||
|
||||
while ((ch = getopt_long(argc, argv, "VmcdvunrIi:b:a:t:g:x:p:", longopts, NULL)) != -1)
|
||||
while ((ch = getopt_long(argc, argv, "VmcdvunlrIi:b:a:t:g:x:p:1:2:o:", longopts, NULL)) != -1)
|
||||
{
|
||||
switch (ch) {
|
||||
case 'm':
|
||||
@@ -1063,13 +1422,28 @@ int main(int argc, char *argv[]) {
|
||||
case 'r':
|
||||
uninstall = 1;
|
||||
break;
|
||||
case '1':
|
||||
bundle_id = optarg;
|
||||
break;
|
||||
case '2':
|
||||
target_filename = optarg;
|
||||
break;
|
||||
case 'o':
|
||||
command_only = true;
|
||||
upload_pathname = optarg;
|
||||
command = "upload";
|
||||
break;
|
||||
case 'l':
|
||||
command_only = true;
|
||||
command = "list";
|
||||
break;
|
||||
default:
|
||||
usage(argv[0]);
|
||||
return exitcode_error;
|
||||
}
|
||||
}
|
||||
|
||||
if (!app_path && !detect_only) {
|
||||
if (!app_path && !detect_only && !command_only) {
|
||||
usage(argv[0]);
|
||||
exit(exitcode_error);
|
||||
}
|
||||
@@ -1103,3 +1477,4 @@ int main(int argc, char *argv[]) {
|
||||
AMDeviceNotificationSubscribe(&device_callback, 0, 0, NULL, ¬ify);
|
||||
CFRunLoopRun();
|
||||
}
|
||||
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "ios-deploy",
|
||||
"version": "1.0.8",
|
||||
"version": "1.0.9",
|
||||
"description": "launch iOS apps iOS devices from the command line (Xcode 5)",
|
||||
"main": "ios-deploy",
|
||||
"scripts": {
|
||||
|
||||
Reference in New Issue
Block a user