diff options
Diffstat (limited to 'src/mobilesync.c')
| -rw-r--r-- | src/mobilesync.c | 585 | 
1 files changed, 585 insertions, 0 deletions
| diff --git a/src/mobilesync.c b/src/mobilesync.c index b5b9453..db437c4 100644 --- a/src/mobilesync.c +++ b/src/mobilesync.c @@ -2,6 +2,7 @@   * mobilesync.c    * Contains functions for the built-in MobileSync client.   *  + * Copyright (c) 2010 Bryan Forbes All Rights Reserved.   * Copyright (c) 2009 Jonathan Beck All Rights Reserved.   *   * This library is free software; you can redistribute it and/or @@ -19,9 +20,14 @@   * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA    */ +#define _GNU_SOURCE 1 +#define __USE_GNU 1 +  #include <plist/plist.h>  #include <string.h>  #include <stdlib.h> +#include <stdio.h> +#include <glib.h>  #include "mobilesync.h"  #include "device_link_service.h" @@ -30,6 +36,8 @@  #define MSYNC_VERSION_INT1 100  #define MSYNC_VERSION_INT2 100 +#define EMPTY_PARAMETER_STRING "___EmptyParameterString___" +  /**   * Convert an device_link_service_error_t value to an mobilesync_error_t value.   * Used internally to get correct error codes when using device_link_service stuff. @@ -84,6 +92,8 @@ mobilesync_error_t mobilesync_client_new(idevice_t device, uint16_t port,  	mobilesync_client_t client_loc = (mobilesync_client_t) malloc(sizeof(struct mobilesync_client_private));  	client_loc->parent = dlclient; +	client_loc->direction = MOBILESYNC_SYNC_DIR_DEVICE_TO_COMPUTER; +	client_loc->data_class = NULL;  	/* perform handshake */  	ret = mobilesync_error(device_link_service_version_exchange(dlclient, MSYNC_VERSION_INT1, MSYNC_VERSION_INT2)); @@ -150,3 +160,578 @@ mobilesync_error_t mobilesync_send(mobilesync_client_t client, plist_t plist)  		return MOBILESYNC_E_INVALID_ARG;  	return mobilesync_error(device_link_service_send(client->parent, plist));  } + +mobilesync_error_t mobilesync_start(mobilesync_client_t client, const char *data_class, mobilesync_anchors_t anchors, mobilesync_sync_type_t *sync_type, uint64_t *data_class_version) +{ +	if (!client || client->data_class || !data_class || +		!anchors || !anchors->computer_anchor) { +		return MOBILESYNC_E_INVALID_ARG; +	} + +	mobilesync_error_t err = MOBILESYNC_E_UNKNOWN_ERROR; +	char *response_type = NULL; +	char *sync_type_str = NULL; +	plist_t msg = NULL; +	plist_t response_type_node = NULL; + +	msg = plist_new_array(); +	plist_array_append_item(msg, plist_new_string("SDMessageSyncDataClassWithDevice")); +	plist_array_append_item(msg, plist_new_string(data_class)); +	if (anchors->device_anchor) { +		plist_array_append_item(msg, plist_new_string(anchors->device_anchor)); +	} else { +		plist_array_append_item(msg, plist_new_string("---")); +	} +	plist_array_append_item(msg, plist_new_string(anchors->computer_anchor)); +	plist_array_append_item(msg, plist_new_uint(*data_class_version)); +	plist_array_append_item(msg, plist_new_string(EMPTY_PARAMETER_STRING)); + +	err = mobilesync_send(client, msg); + +	if (err != MOBILESYNC_E_SUCCESS) { +		goto out; +	} + +	plist_free(msg); +	msg = NULL; + +	err = mobilesync_receive(client, &msg); + +	if (err != MOBILESYNC_E_SUCCESS) { +		goto out; +	} + +	response_type_node = plist_array_get_item(msg, 0); +	if (!response_type_node) { +		err = MOBILESYNC_E_PLIST_ERROR; +		goto out; +	} + +	plist_get_string_val(response_type_node, &response_type); +	if (!response_type) { +		err = MOBILESYNC_E_PLIST_ERROR; +		goto out; +	} + +	if (!strcmp(response_type, "SDMessageRefuseToSyncDataClassWithComputer")) { +		char *reason = NULL; +		err = MOBILESYNC_E_SYNC_REFUSED; +		plist_get_string_val(plist_array_get_item(msg, 2), &reason); +		debug_info("Device refused sync: %s", reason); +		free(reason); +		reason = NULL; +		goto out; +	} + +	if (!strcmp(response_type, "SDMessageCancelSession")) { +		char *reason = NULL; +		err = MOBILESYNC_E_CANCELLED; +		plist_get_string_val(plist_array_get_item(msg, 2), &reason); +		debug_info("Device cancelled: %s", reason); +		free(reason); +		reason = NULL; +		goto out; +	} + +	if (sync_type != NULL) { +		plist_t sync_type_node = plist_array_get_item(msg, 4); +		if (!sync_type_node) { +			err = MOBILESYNC_E_PLIST_ERROR; +			goto out; +		} + +		plist_get_string_val(sync_type_node, &sync_type_str); +		if (!sync_type_str) { +			err = MOBILESYNC_E_PLIST_ERROR; +			goto out; +		} + +		if (!strcmp(sync_type_str, "SDSyncTypeFast")) { +			*sync_type = MOBILESYNC_SYNC_TYPE_FAST; +		} else if (!strcmp(sync_type_str, "SDSyncTypeSlow")) { +			*sync_type = MOBILESYNC_SYNC_TYPE_SLOW; +		} else if (!strcmp(sync_type_str, "SDSyncTypeReset")) { +			*sync_type = MOBILESYNC_SYNC_TYPE_RESET; +		} else { +			err = MOBILESYNC_E_PLIST_ERROR; +			goto out; +		} +	} + +	if (data_class_version != NULL) { +		plist_t data_class_version_node = plist_array_get_item(msg, 5); +		if (!data_class_version_node) { +			err = MOBILESYNC_E_PLIST_ERROR; +			goto out; +		} + +		plist_get_uint_val(data_class_version_node, data_class_version); +	} + +	err = MOBILESYNC_E_SUCCESS; + +	out: +	if (sync_type_str) { +		free(sync_type_str); +		sync_type_str = NULL; +	} +	if (response_type) { +		free(response_type); +		response_type = NULL; +	} +	if (msg) { +		plist_free(msg); +		msg = NULL; +	} + +	client->data_class = strdup(data_class); +	client->direction = MOBILESYNC_SYNC_DIR_DEVICE_TO_COMPUTER; +	return err; +} + +mobilesync_error_t mobilesync_finish(mobilesync_client_t client) +{ +	if (!client || !client->data_class) { +		return MOBILESYNC_E_INVALID_ARG; +	} + +	mobilesync_error_t err = MOBILESYNC_E_UNKNOWN_ERROR; + +	plist_t msg = NULL; +	plist_t response_type_node = NULL; +	char *response_type = NULL; + +	msg = plist_new_array(); +	plist_array_append_item(msg, plist_new_string("SDMessageFinishSessionOnDevice")); +	plist_array_append_item(msg, plist_new_string(client->data_class)); + +	err = mobilesync_send(client, msg); + +	if (err != MOBILESYNC_E_SUCCESS) { +		goto out; +	} + +	plist_free(msg); +	msg = NULL; + +	err = mobilesync_receive(client, &msg); + +	if (err != MOBILESYNC_E_SUCCESS) { +		goto out; +	} + +	response_type_node = plist_array_get_item(msg, 0); +	if (!response_type_node) { +		err = MOBILESYNC_E_PLIST_ERROR; +		goto out; +	} + +	plist_get_string_val(response_type_node, &response_type); +	if (!response_type) { +		err = MOBILESYNC_E_PLIST_ERROR; +		goto out; +	} + +	if (!strcmp(response_type, "SDMessageDeviceFinishedSession")) { +		err = MOBILESYNC_E_SUCCESS; +	} + +	out: +	if (response_type) { +		free(response_type); +		response_type = NULL; +	} +	if (msg) { +		plist_free(msg); +		msg = NULL; +	} + +	free(client->data_class); +	client->data_class = NULL; +	client->direction = MOBILESYNC_SYNC_DIR_DEVICE_TO_COMPUTER; +	return err; +} + +static mobilesync_error_t mobilesync_get_records(mobilesync_client_t client, const char *operation) +{ +	if (!client || !client->data_class || !operation) { +		return MOBILESYNC_E_INVALID_ARG; +	} + +	mobilesync_error_t err = MOBILESYNC_E_UNKNOWN_ERROR; +	plist_t msg = NULL; + +	msg = plist_new_array(); +	plist_array_append_item(msg, plist_new_string(operation)); +	plist_array_append_item(msg, plist_new_string(client->data_class)); +	 +	err = mobilesync_send(client, msg); + +	if (msg) { +		plist_free(msg); +		msg = NULL; +	} +	return err; +} + +mobilesync_error_t mobilesync_get_all_records_from_device(mobilesync_client_t client) +{ +	return mobilesync_get_records(client, "SDMessageGetAllRecordsFromDevice"); +} + +mobilesync_error_t mobilesync_get_changes_from_device(mobilesync_client_t client) +{ +	return mobilesync_get_records(client, "SDMessageGetChangesFromDevice"); +} + +mobilesync_error_t mobilesync_receive_changes(mobilesync_client_t client, plist_t *entities, uint8_t *is_last_record, plist_t *actions) +{ +	if (!client || !client->data_class) { +		return MOBILESYNC_E_INVALID_ARG; +	} + +	plist_t msg = NULL; +	plist_t response_type_node = NULL; +	plist_t actions_node = NULL; +	char *response_type = NULL; +	uint8_t has_more_changes = 0; + +	mobilesync_error_t err = mobilesync_receive(client, &msg); +	if (err != MOBILESYNC_E_SUCCESS) { +		goto out; +	} + +	response_type_node = plist_array_get_item(msg, 0); +	if (!response_type_node) { +		err = MOBILESYNC_E_PLIST_ERROR; +		goto out; +	} + +	plist_get_string_val(response_type_node, &response_type); +	if (!response_type) { +		err = MOBILESYNC_E_PLIST_ERROR; +		goto out; +	} + +	if (!strcmp(response_type, "SDMessageCancelSession")) { +		char *reason = NULL; +		err = MOBILESYNC_E_CANCELLED; +		plist_get_string_val(plist_array_get_item(msg, 2), &reason); +		debug_info("Device cancelled: %s", reason); +		free(reason); +		goto out; +	} + +	*entities = plist_copy(plist_array_get_item(msg, 2)); + +	plist_get_bool_val(plist_array_get_item(msg, 3), &has_more_changes); +	*is_last_record = (has_more_changes > 0 ? 0 : 1); + +	actions_node = plist_array_get_item(msg, 4); +	if (plist_get_node_type(actions) == PLIST_DICT) +		*actions = plist_copy(actions_node); +	else +		*actions = NULL; + +	out: +	if (response_type) { +		free(response_type); +		response_type = NULL; +	} +	if (msg) { +		plist_free(msg); +		msg = NULL; +	} +	return err; +} + +mobilesync_error_t mobilesync_acknowledge_changes_from_device(mobilesync_client_t client) +{ +	if (!client || !client->data_class) { +		return MOBILESYNC_E_INVALID_ARG; +	} + +	plist_t msg = NULL; +	mobilesync_error_t err = MOBILESYNC_E_UNKNOWN_ERROR; + +	msg = plist_new_array(); +	plist_array_append_item(msg, plist_new_string("SDMessageAcknowledgeChangesFromDevice")); +	plist_array_append_item(msg, plist_new_string(client->data_class)); + +	err = mobilesync_send(client, msg); +	plist_free(msg); +	return err; +} + +static plist_t create_process_changes_message(const char *data_class, plist_t entities, uint8_t more_changes, plist_t actions) +{ +	plist_t msg = plist_new_array(); +	plist_array_append_item(msg, plist_new_string("SDMessageProcessChanges")); +	plist_array_append_item(msg, plist_copy(entities)); +	plist_array_append_item(msg, plist_new_bool(more_changes)); +	plist_array_append_item(msg, plist_copy(actions)); + +	return msg; +} + +mobilesync_error_t mobilesync_ready_to_send_changes_from_computer(mobilesync_client_t client) +{ +	if (!client || !client->data_class) { +		return MOBILESYNC_E_INVALID_ARG; +	} + +	if (client->direction != MOBILESYNC_SYNC_DIR_DEVICE_TO_COMPUTER) { +		return MOBILESYNC_E_WRONG_DIRECTION; +	} + +	plist_t msg = NULL; +	plist_t response_type_node = NULL; +	char *response_type = NULL; +	mobilesync_error_t err = MOBILESYNC_E_UNKNOWN_ERROR; + +	err = mobilesync_receive(client, &msg); +	if (err != MOBILESYNC_E_SUCCESS) { +		goto out; +	} + +	response_type_node = plist_array_get_item(msg, 0); +	if (!response_type_node) { +		err = MOBILESYNC_E_PLIST_ERROR; +		goto out; +	} + +	plist_get_string_val(response_type_node, &response_type); +	if (!response_type) { +		err = MOBILESYNC_E_PLIST_ERROR; +		goto out; +	} + +	if (!strcmp(response_type, "SDMessageCancelSession")) { +		char *reason = NULL; +		err = MOBILESYNC_E_CANCELLED; +		plist_get_string_val(plist_array_get_item(msg, 2), &reason); +		debug_info("Device cancelled: %s", reason); +		free(reason); +		goto out; +	} + +	if (strcmp(response_type, "SDMessageDeviceReadyToReceiveChanges") != 0) { +		err = MOBILESYNC_E_NOT_READY; +		goto out; +	} + +	err = mobilesync_error(device_link_service_send_ping(client->parent, "Preparing to get changes for device")); +	if (err != MOBILESYNC_E_SUCCESS) { +		goto out; +	} + +	client->direction = MOBILESYNC_SYNC_DIR_COMPUTER_TO_DEVICE; +	err = MOBILESYNC_E_SUCCESS; + +	out: +	if (msg) { +		plist_free(msg); +		msg = NULL; +	} + +	return err; +} + +mobilesync_error_t mobilesync_send_changes(mobilesync_client_t client, plist_t entities, uint8_t is_last_record, plist_t actions) +{ +	if (!client || !client->data_class || !entities) { +		return MOBILESYNC_E_INVALID_ARG; +	} + +	if (plist_get_node_type(entities) != PLIST_DICT) { +		return MOBILESYNC_E_INVALID_ARG; +	} + +	if (client->direction != MOBILESYNC_SYNC_DIR_COMPUTER_TO_DEVICE) { +		return MOBILESYNC_E_WRONG_DIRECTION; +	} + +	mobilesync_error_t err = MOBILESYNC_E_UNKNOWN_ERROR; +	plist_t msg = NULL; + +	msg = create_process_changes_message(client->data_class, entities, (is_last_record > 0 ? 0 : 1), actions); +	err = mobilesync_send(client, msg); + +	if (msg) { +		plist_free(msg); +		msg = NULL; +	} + +	return err; +} + +mobilesync_error_t mobilesync_remap_identifiers(mobilesync_client_t client, plist_t *mapping) +{ +	if (!client || !client->data_class) { +		return MOBILESYNC_E_INVALID_ARG; +	} + +	if (client->direction == MOBILESYNC_SYNC_DIR_DEVICE_TO_COMPUTER) { +		return MOBILESYNC_E_WRONG_DIRECTION; +	} + +	plist_t msg = NULL; +	plist_t response_type_node = NULL; +	char *response_type = NULL; + +	mobilesync_error_t err = mobilesync_receive(client, &msg); +	if (err != MOBILESYNC_E_SUCCESS) { +		goto out; +	} + +	response_type_node = plist_array_get_item(msg, 0); +	if (!response_type_node) { +		err = MOBILESYNC_E_PLIST_ERROR; +		goto out; +	} + +	plist_get_string_val(response_type_node, &response_type); +	if (!response_type) { +		err = MOBILESYNC_E_PLIST_ERROR; +		goto out; +	} + +	if (!strcmp(response_type, "SDMessageCancelSession")) { +		char *reason = NULL; +		err = MOBILESYNC_E_CANCELLED; +		plist_get_string_val(plist_array_get_item(msg, 2), &reason); +		debug_info("Device cancelled: %s", reason); +		free(reason); +		goto out; +	} + +	if (strcmp(response_type, "SDMessageRemapRecordIdentifiers") != 0) { +		err = MOBILESYNC_E_PLIST_ERROR; +		goto out; +	} + +	if (mapping != NULL) { +		plist_t map = plist_array_get_item(msg, 2); +		if (plist_get_node_type(map) == PLIST_DICT) { +			*mapping = plist_copy(map); +		} else { +			*mapping = NULL; +		} +	} + +	err = MOBILESYNC_E_SUCCESS; + +	out: +	if (response_type) { +		free(response_type); +		response_type = NULL; +	} +	if (msg) { +		plist_free(msg); +		msg = NULL; +	} + +	return err; +} + +mobilesync_error_t mobilesync_cancel(mobilesync_client_t client, const char* reason) +{ +	if (!client || !client->data_class || !reason) { +		return MOBILESYNC_E_INVALID_ARG; +	} + +	mobilesync_error_t err = MOBILESYNC_E_UNKNOWN_ERROR; +	plist_t msg = NULL; + +	msg = plist_new_array(); +	plist_array_append_item(msg, plist_new_string("SDMessageCancelSession")); +	plist_array_append_item(msg, plist_new_string(client->data_class)); +	plist_array_append_item(msg, plist_new_string(reason)); + +	err = mobilesync_send(client, msg); + +	plist_free(msg); +	msg = NULL; + +	free(client->data_class); +	client->data_class = NULL; +	client->direction = MOBILESYNC_SYNC_DIR_DEVICE_TO_COMPUTER; + +	return err; +} + +mobilesync_anchors_t mobilesync_anchors_new(const char *device_anchor, const char *computer_anchor) +{ +	mobilesync_anchors_t anchors = (mobilesync_anchors_t) malloc(sizeof(mobilesync_anchors));  +	if (device_anchor != NULL) { +		anchors->device_anchor = strdup(device_anchor); +	} else { +		anchors->device_anchor = NULL; +	} +	if (computer_anchor != NULL) { +		anchors->computer_anchor = strdup(computer_anchor); +	} else { +		anchors->computer_anchor = NULL; +	} + +	return anchors; +} + +void mobilesync_anchors_free(mobilesync_anchors_t anchors) +{ +	if (anchors->device_anchor != NULL) { +		free(anchors->device_anchor); +		anchors->device_anchor = NULL; +	} +	if (anchors->computer_anchor != NULL) { +		free(anchors->computer_anchor); +		anchors->computer_anchor = NULL; +	} +	free(anchors); +	anchors = NULL; +} + +plist_t mobilesync_actions_new() +{ +	return plist_new_dict(); +} + +void mobilesync_actions_add(plist_t actions, ...) +{ +	if (!actions) +		return; +	va_list args; +	va_start(args, actions); +	char *arg = va_arg(args, char*); +	while (arg) { +		char *key = strdup(arg); +		if (!strcmp(key, "SyncDeviceLinkEntityNamesKey")) { +			char **entity_names = va_arg(args, char**); +			int entity_names_length = va_arg(args, int); +			int i = 0; + +			plist_t array = plist_new_array(); + +			for (i = 0; i < entity_names_length; i++) { +				plist_array_append_item(array, plist_new_string(entity_names[i])); +			} + +			plist_dict_insert_item(actions, key, array); +		} else if (!strcmp(key, "SyncDeviceLinkAllRecordsOfPulledEntityTypeSentKey")) { +			int link_records = va_arg(args, int); +			plist_dict_insert_item(actions, key, plist_new_bool(link_records)); +		} +		free(key); +		key = NULL; +		arg = va_arg(args, char*); +	} +	va_end(args); +} + +void mobilesync_actions_free(plist_t actions) +{ +	if (actions) { +		plist_free(actions); +		actions = NULL; +	} +} | 
