summaryrefslogtreecommitdiffstats
path: root/tests/clients/ldifsort.c
blob: fcd7923d2ff9ea02a6be60ba4fbeafe5537aeccf (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
/*
 * Copyright 2008 Red Hat, Inc.
 *
 * 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; version 2 of the License.
 *
 * 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, write to the
 *
 *   Free Software Foundation, Inc.
 *   59 Temple Place, Suite 330
 *   Boston, MA 02111-1307 USA
 *
 */

#include "../../src/config.h"
#include <sys/types.h>
#include <limits.h>
#include <search.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#define MAX_ENTRIES 4096

struct ldif_entry {
	char *dn;
	char *entry;
};

static int
compare_entries(const void *a, const void *b)
{
	return strcoll(((const struct ldif_entry *) a)->dn,
		       ((const struct ldif_entry *) b)->dn);
}

int
main(int argc, char **argv)
{
	FILE *infile;
	char buf[LINE_MAX], *p;
	struct ldif_entry entry, entries[MAX_ENTRIES];
	size_t n_entries, l;
	memset(&entry, 0, sizeof(entry));
	memset(&entries, 0, sizeof(entries));
	n_entries = 0;
	infile = (argc > 1) ? fopen(argv[1], "r") : stdin;
	while ((infile != NULL) && (fgets(buf, sizeof(buf), infile) != NULL)) {
		if (buf[0] == '#') {
			continue;
		}
		p = strchr(buf, '\n');
		if (p != NULL) {
			if ((p == buf) && (entry.dn != NULL)) {
				if (n_entries <
				    sizeof(entries) / sizeof(entries[0])) {
					lsearch(&entry,
						(void *) &entries,
						&n_entries,
						sizeof(entry),
						&compare_entries);
				}
				memset(&entry, 0, sizeof(entry));
			}
			if (strncasecmp(buf, "dn:", 3) == 0) {
				*p = '\0';
				entry.dn = strdup(buf);
				entry.entry = NULL;
			} else {
				l = (entry.entry ? strlen(entry.entry) : 0) +
				    strlen(buf) + 1;
				p = malloc(l);
				if (p != NULL) {
					if (entry.entry) {
						strcpy(p, entry.entry);
						strcat(p, buf);
					} else {
						strcpy(p, buf);
					}
					free(entry.entry);
					entry.entry = p;
				}
			}
		}
	}
	qsort((void *) &entries, n_entries, sizeof(entry), &compare_entries);
	for (l = 0; l < n_entries; l++) {
		puts(entries[l].dn);
		if (entries[l].entry != NULL) {
			puts(entries[l].entry);
		} else {
			puts("");
		}
	}
	return 0;
}