/database/subscription.c
/* Blerg is (C) 2011 The Dominion of Awesome, and is distributed under a
 * BSD-style license.  Please see the COPYING file for details.
 */
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/mman.h>
#include <stdlib.h>
#include <unistd.h>
#include <stdio.h>
#include <fcntl.h>
#include <string.h>
#include "subscription.h"
#include "stringbucket.h"
#include "util.h"
#include "config.h"
#include "configuration.h"

int subscription_add(const char *from, const char *to) {
	char filename[FILENAME_MAX];
	struct stringbucket * sb;

	snprintf(filename, FILENAME_MAX, "%s/%s/subscriptions", blergconf.data_path, from);
	sb = stringbucket_open(filename);
	stringbucket_add(sb, to);
	stringbucket_close(sb);

	snprintf(filename, FILENAME_MAX, "%s/%s/subscribers", blergconf.data_path, to);
	sb = stringbucket_open(filename);
	stringbucket_add(sb, from);
	stringbucket_close(sb);

	return 1;
}

int subscription_remove(const char *from, const char *to) {
	char filename[FILENAME_MAX];
	struct stringbucket * sb;

	snprintf(filename, FILENAME_MAX, "%s/%s/subscriptions", blergconf.data_path, from);
	sb = stringbucket_open(filename);
	stringbucket_delete(sb, to);
	stringbucket_close(sb);

	snprintf(filename, FILENAME_MAX, "%s/%s/subscribers", blergconf.data_path, to);
	sb = stringbucket_open(filename);
	stringbucket_delete(sb, from);
	stringbucket_close(sb);

	return 1;
}

void subscription_notify_add_item(char *to, void *stuff) {
	char filename[FILENAME_MAX];

	snprintf(filename, FILENAME_MAX, "%s/%s/subscription_feed", blergconf.data_path, to);
	int fd = open(filename, O_WRONLY | O_APPEND | O_CREAT, 0600);
	write(fd, stuff, sizeof(struct blergref));
	close(fd);
}

int subscription_notify(const char *author, uint64_t record) {
	char filename[FILENAME_MAX];
	struct blergref r;

	strncpy(r.author, author, 32);
	r.record = record;

	snprintf(filename, FILENAME_MAX, "%s/%s/subscribers", blergconf.data_path, author);
	struct stringbucket * sb = stringbucket_open(filename);
	stringbucket_iterate(sb, subscription_notify_add_item, &r);
	stringbucket_close(sb);

	return 1;
}

struct blergref * subscription_list(const char *author, uint64_t offset, int *count, int direction) {
	char filename[FILENAME_MAX];
	struct stat st;
	struct blergref * slist;
	struct blergref * retlist;
	uint64_t n_subscription_records;

	if (!valid_name(author))
		return NULL;

	snprintf(filename, FILENAME_MAX, "%s/%s/subscription_feed", blergconf.data_path, author);

	int feed_fd = open(filename, O_RDONLY);
	if (feed_fd == -1) {
		perror("Could not open subscription feed");
		goto subscription_list_open_failed;
	}

	fstat(feed_fd, &st);
	if (st.st_size == 0) {
		fprintf(stderr, "Feed is empty\n");
		goto subscription_list_map_failed;
	}
	n_subscription_records = st.st_size / sizeof(struct blergref);
	if (*count > n_subscription_records - offset)
		*count = n_subscription_records - offset;
	if (offset > n_subscription_records) {
		fprintf(stderr, "Cannot access subscription record beyond end\n");
		goto subscription_list_map_failed;
		close(feed_fd);
		return NULL;
	}

	slist = (struct blergref *) mmap(NULL, st.st_size, PROT_READ, MAP_SHARED, feed_fd, 0);
	if (slist == MAP_FAILED) {
		perror("Could not mmap tag_file");
		goto subscription_list_map_failed;
	}
	retlist = (struct blergref *) malloc(sizeof(struct blergref) * *count);
	if (retlist == NULL) {
		perror("Could not allocate memory for subscription feed list");
		goto subscription_list_malloc_failed;
	}
	
	switch(direction) {
	case 1:
		memcpy(retlist, slist + offset, sizeof(struct blergref) * *count);
		break;
	case -1:
		memcpy(retlist, slist + (n_subscription_records - *count - offset), sizeof(struct blergref) * *count);
	}

	munmap(slist, st.st_size);
	close(feed_fd);
	return retlist;

subscription_list_malloc_failed:
	munmap(slist, st.st_size);
subscription_list_map_failed:
	close(feed_fd);
subscription_list_open_failed:
	*count = 0;
	return NULL;
}

int is_subscribed(const char *from, const char *to) {
	char filename[FILENAME_MAX];
	struct stringbucket * sb;
	int ret = 0;

	snprintf(filename, FILENAME_MAX, "%s/%s/subscriptions", blergconf.data_path, from);
	sb = stringbucket_open(filename);
	if (stringbucket_find(sb, to) != -1)
		ret = 1;
	stringbucket_close(sb);

	return ret;
}

int subscription_count_items(const char *user) {
	char filename[FILENAME_MAX];
	struct stat st;

	if (!valid_name(user))
		return -1;

	snprintf(filename, FILENAME_MAX, "%s/%s/subscription_feed", blergconf.data_path, user);

	if (access(filename, R_OK) != 0)
		return 0;
	stat(filename, &st);
	return st.st_size / sizeof(struct blergref);
}