aboutsummaryrefslogblamecommitdiffstats
path: root/set-contact-addresses.c
blob: 7606e02e993dfcfbaec87e7ce8d56ef6fbbefadc (plain) (tree)




















                                                                                                                

                                                                                   







































































































































































































                                                                                                                                
                                                                               



















                                                                                     
/* See LICENSE file for copyright and license details. */
#include "common.h"

USAGE("[-a address | -A address] [-c context | -C context] [-g [latitude]:[longitude] | -G latitude:longitude] "
      "[-n country | -N country] [-o care-of | -O care-of] [-p post-code | -P post-code] [-t city | -T city] "
      "[-u] contact-id");


static int
size_t_cmp(const void *av, const void *bv)
{
	const size_t *ap = av, *bp = bv;
	return *ap < *bp ? -1 : *ap > *bp;
}

int
main(int argc, char *argv[])
{
	int edit_address = 0, edit_context = 0, edit_location = 0;
	int edit_country = 0, edit_careof = 0, edit_postcode = 0;
	int edit_city = 0, edit = 0, remove = 0;
	const char *address = NULL, *context = NULL, *location = NULL;
	const char *country = NULL, *careof = NULL, *postcode = NULL, *city = NULL;
	double lat, lon, lat_min = 0, lat_max = 0, lon_min = 0, lon_max = 0, flat, flon;
	struct passwd *user;
	struct libcontacts_contact contact;
	size_t i, *indices = NULL, nindices, naddresses;
	char *p;

	ARGBEGIN {
	case 'C':
		edit_context = 1;
		edit = 1;
		/* fall through */
	case 'c':
		if (context)
			usage();
		context = ARG();
		break;
	case 'O':
		edit_careof = 1;
		edit = 1;
		/* fall through */
	case 'o':
		if (careof)
			usage();
		careof = ARG();
		break;
	case 'A':
		edit_address = 1;
		edit = 1;
		/* fall through */
	case 'a':
		if (address)
			usage();
		address = ARG();
		break;
	case 'P':
		edit_postcode = 1;
		edit = 1;
		/* fall through */
	case 'p':
		if (postcode)
			usage();
		postcode = ARG();
		break;
	case 'T':
		edit_city = 1;
		edit = 1;
		/* fall through */
	case 't':
		if (city)
			usage();
		city = ARG();
		break;
	case 'N':
		edit_country = 1;
		edit = 1;
		/* fall through */
	case 'n':
		if (country)
			usage();
		country = ARG();
		break;
	case 'G':
		edit_location = 1;
		edit = 1;
		/* fall through */
	case 'g':
		if (location)
			usage();
		location = ARG();
		if (edit_location) {
			p = strchr(location, ':');
			if (!p || !p[1] || p == location)
				usage();
		}
		if (parse_coord(location, &lat, &lat_min, &lat_max, &lon, &lon_min, &lon_max))
			usage();
		break;
	case 'u':
		remove = 1;
		break;
	default:
		usage();
	} ARGEND;

	if (remove == edit) {
		if (edit)
			eprintf("-u cannot be combined with -ACGLOPT\n");
		eprintf("at least one of -ACGLOPTu is required\n");
	}

	if ((!context  || edit_context) && (!careof   || edit_careof)   &&
	    (!address  || edit_address) && (!postcode || edit_postcode) &&
	    (!city     || edit_city)    && (!country  || edit_country)  &&
	    (!location || edit_location))
		edit = 0;

	if (argc != 1 || !*argv[0] || strchr(argv[0], '/'))
		usage();

	errno = 0;
	user = getpwuid(getuid());
	if (!user)
		eprintf("getpwuid: %s\n", errno ? strerror(errno) : "user does not exist");

	if (libcontacts_load_contact(argv[0], &contact, user))
		eprintf("libcontacts_load_contact %s: %s\n", argv[0], errno ? strerror(errno) : "contact file is malformatted");

	naddresses = 0;
	if (contact.addresses)
		for (; contact.addresses[naddresses]; naddresses++);

	if ((edit || remove) && naddresses) {
		nindices = naddresses;
		indices = emalloc(nindices * sizeof(*contact.addresses));
		for (i = 0; i < nindices; i++)
			indices[i] = i;

		if (context && !edit_context)
			for (i = 0; i < nindices;)
				if (strcmpnul(contact.addresses[indices[i++]]->context, context))
					indices[--i] = indices[--nindices];
		if (careof && !edit_careof)
			for (i = 0; i < nindices;)
				if (strcmpnul(contact.addresses[indices[i++]]->care_of, careof))
					indices[--i] = indices[--nindices];
		if (address && !edit_address)
			for (i = 0; i < nindices;)
				if (strcmpnul(contact.addresses[indices[i++]]->address, address))
					indices[--i] = indices[--nindices];
		if (postcode && !edit_postcode)
			for (i = 0; i < nindices;)
				if (strcmpnul(contact.addresses[indices[i++]]->postcode, postcode))
					indices[--i] = indices[--nindices];
		if (city && !edit_city)
			for (i = 0; i < nindices;)
				if (strcmpnul(contact.addresses[indices[i++]]->city, city))
					indices[--i] = indices[--nindices];
		if (country && !edit_country)
			for (i = 0; i < nindices;)
				if (strcmpnul(contact.addresses[indices[i++]]->country, country))
					indices[--i] = indices[--nindices];
		if (location && !edit_location) {
			for (i = 0; i < nindices;) {
				if (!contact.addresses[indices[i++]]->have_coordinates) {
					flat = contact.addresses[indices[i - 1]]->latitude;
					flon = contact.addresses[indices[i - 1]]->longitude;
					if (!(lat_min <= flat && flat <= lat_max && lon_min <= flon && flon <= lon_max))
						indices[--i] = indices[--nindices];
				}
			}
		}

		if (edit) {
			while (nindices--) {
				i = indices[nindices];
				if (edit_context) {
					free(contact.addresses[i]->context);
					contact.addresses[i]->context = estrdup(context);
				}
				if (edit_careof) {
					free(contact.addresses[i]->care_of);
					contact.addresses[i]->care_of = estrdup(careof);
				}
				if (edit_address) {
					free(contact.addresses[i]->address);
					contact.addresses[i]->address = estrdup(address);
				}
				if (edit_postcode) {
					free(contact.addresses[i]->postcode);
					contact.addresses[i]->postcode = estrdup(postcode);
				}
				if (edit_city) {
					free(contact.addresses[i]->city);
					contact.addresses[i]->city = estrdup(city);
				}
				if (edit_country) {
					free(contact.addresses[i]->country);
					contact.addresses[i]->country = estrdup(country);
				}
				if (edit_location) {
					contact.addresses[i]->latitude = lat;
					contact.addresses[i]->longitude = lon;
					contact.addresses[i]->have_coordinates = 1;
				}
			}
		} else {
			qsort(indices, nindices, sizeof(*indices), size_t_cmp);
			while (nindices--) { /* reverse order is important */
				i = indices[nindices];
				libcontacts_address_destroy(contact.addresses[i]);
				free(contact.addresses[i]);
				/* we using memmove for simplity as we only expect to remove one entry */
				memmove(&contact.addresses[i], &contact.addresses[i + 1],
				        (naddresses-- - i) * sizeof(*contact.addresses));
			}
		}
	} else if (!edit && !remove) {
		i = naddresses;
		contact.addresses = erealloc(contact.addresses, (i + 2) * sizeof(*contact.addresses));
		contact.addresses[i + 1] = NULL;
		contact.addresses[i] = ecalloc(1, sizeof(**contact.addresses));
		contact.addresses[i]->context  = context  ? estrdup(context)  : NULL;
		contact.addresses[i]->care_of  = careof   ? estrdup(careof)   : NULL;
		contact.addresses[i]->address  = address  ? estrdup(address)  : NULL;
		contact.addresses[i]->postcode = postcode ? estrdup(postcode) : NULL;
		contact.addresses[i]->city     = city     ? estrdup(city)     : NULL;
		contact.addresses[i]->country  = country  ? estrdup(country)  : NULL;
		if (location) {
			contact.addresses[i]->have_coordinates = 1;
			contact.addresses[i]->latitude = lat;
			contact.addresses[i]->longitude = lon;
		}
	}

	if (libcontacts_save_contact(&contact, user))
		eprintf("libcontacts_save_contact %s:", argv[0]);
	libcontacts_contact_destroy(&contact);
	free(indices);

	return 0;
}