5 Author: Pekka Riikonen <priikone@silcnet.org>
7 Copyright (C) 2002 Pekka Riikonen
9 This program is free software; you can redistribute it and/or modify
10 it under the terms of the GNU General Public License as published by
11 the Free Software Foundation; version 2 of the License.
13 This program is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU General Public License for more details.
20 /* Implementation of the VCard (RFC 2426) */
22 #include "silcincludes.h"
24 #define VCARD_HEADER "BEGIN:VCARD\n"
25 #define VCARD_VERSION "VERSION:3.0\n"
26 #define VCARD_FOOTER "END:VCARD"
30 unsigned char *silc_vcard_encode(SilcVCard vcard, SilcUInt32 *vcard_len)
32 SilcBufferStruct buffer;
35 if (!vcard->full_name || !vcard->family_name || !vcard->first_name)
38 memset(&buffer, 0, sizeof(buffer));
39 silc_buffer_strformat(
43 "FN:", vcard->full_name, "\n",
44 "N:", vcard->family_name, ";", vcard->first_name, ";",
45 vcard->middle_names, ";", vcard->prefix, ";", vcard->suffix, "\n",
49 silc_buffer_strformat(&buffer,
50 "NICKNAME:", vcard->nickname, "\n",
53 silc_buffer_strformat(&buffer,
54 "BDAY:", vcard->bday, "\n",
57 silc_buffer_strformat(&buffer,
58 "TITLE:", vcard->title, "\n",
61 silc_buffer_strformat(&buffer,
62 "ROLE:", vcard->role, "\n",
65 silc_buffer_strformat(&buffer,
66 "ORG:", vcard->org_name, ";", vcard->org_unit, "\n",
68 if (vcard->categories)
69 silc_buffer_strformat(&buffer,
70 "CATEGORIES:", vcard->categories, "\n",
73 silc_buffer_strformat(&buffer,
74 "CLASS:", vcard->class, "\n",
77 silc_buffer_strformat(&buffer,
78 "URL:", vcard->url, "\n",
81 silc_buffer_strformat(&buffer,
82 "LABEL;", vcard->url, "\n",
84 for (i = 0; i < vcard->num_addrs; i++) {
85 silc_buffer_strformat(&buffer,
87 vcard->addrs[i].type, ":",
88 vcard->addrs[i].pbox, ";",
89 vcard->addrs[i].ext_addr, ";",
90 vcard->addrs[i].street_addr, ";",
91 vcard->addrs[i].city, ";",
92 vcard->addrs[i].state, ";",
93 vcard->addrs[i].code, ";",
94 vcard->addrs[i].country, "\n",
97 for (i = 0; i < vcard->num_tels; i++) {
98 silc_buffer_strformat(&buffer,
100 vcard->tels[i].type, ":",
101 vcard->tels[i].tel, "\n",
104 for (i = 0; i < vcard->num_emails; i++) {
105 silc_buffer_strformat(&buffer,
107 vcard->emails[i].type, ":",
108 vcard->emails[i].address, "\n",
112 silc_buffer_strformat(&buffer,
113 "NOTE:", vcard->note, "\n",
116 silc_buffer_strformat(&buffer,
117 "REV:", vcard->rev, "\n",
120 silc_buffer_strformat(&buffer, VCARD_FOOTER, SILC_STR_END);
123 *vcard_len = buffer.truelen;
129 #define VCARD_TOKEN(x) \
131 (x) = silc_memdup(val + off, i - off); \
136 /* Take on TYPE= token and prepare for next token */
137 #define VCARD_TYPETOKEN(x) \
140 (x) = silc_memdup(val + off + 5, i - off - 5 - 1); \
141 tmpi = off + 5 + strlen((x)) + 1; \
146 /* Take last token */
147 #define VCARD_LASTTOKEN(x) \
150 (x) = silc_memdup(val + off, len - off); \
155 bool silc_vcard_decode(const unsigned char *data, SilcUInt32 data_len,
159 bool has_begin = FALSE, has_end = FALSE;
162 val = (unsigned char *)data;
164 if (!strchr(val, '\n'))
166 len = strchr(val, '\n') - (char *)val;
167 if (len > data_len - (val - data))
170 if (!strncasecmp(val, VCARD_HEADER, strlen(VCARD_HEADER))) {
172 } else if (!strncasecmp(val, "FN:", 3)) {
173 if (vcard->full_name)
176 vcard->full_name = silc_memdup(val + 3, len - 3);
177 } else if (!strncasecmp(val, "N:", 2)) {
178 if (vcard->family_name)
182 for (i = off; i < len; i++)
184 VCARD_TOKEN(vcard->family_name);
185 VCARD_TOKEN(vcard->first_name);
186 VCARD_TOKEN(vcard->middle_names);
187 VCARD_TOKEN(vcard->prefix);
189 VCARD_LASTTOKEN(vcard->suffix);
191 } else if (!strncasecmp(val, "NICKNAME:", 9)) {
195 vcard->nickname = silc_memdup(val + 9, len - 9);
196 } else if (!strncasecmp(val, "BDAY:", 5)) {
200 vcard->bday = silc_memdup(val + 5, len - 5);
201 } else if (!strncasecmp(val, "TITLE:", 6)) {
205 vcard->title = silc_memdup(val + 6, len - 6);
206 } else if (!strncasecmp(val, "ROLE:", 5)) {
210 vcard->role = silc_memdup(val + 5, len - 5);
211 } else if (!strncasecmp(val, "ORG:", 4)) {
216 for (i = off; i < len; i++) {
218 VCARD_TOKEN(vcard->org_name);
222 /* It's possible to have ORG without last ';', so check for it */
223 if (!vcard->org_name) {
224 VCARD_LASTTOKEN(vcard->org_name);
226 VCARD_LASTTOKEN(vcard->org_unit);
229 } else if (!strncasecmp(val, "CATEGORIES:", 11)) {
230 if (vcard->categories)
233 vcard->categories = silc_memdup(val + 11, len - 11);
234 } else if (!strncasecmp(val, "CLASS:", 6)) {
238 vcard->class = silc_memdup(val + 6, len - 6);
239 } else if (!strncasecmp(val, "URL:", 4)) {
243 vcard->url = silc_memdup(val + 4, len - 4);
244 } else if (!strncasecmp(val, "LABEL;", 6)) {
248 vcard->label = silc_memdup(val + 6, len - 6);
249 } else if (!strncasecmp(val, "ADR;", 4)) {
250 vcard->addrs = silc_realloc(vcard->addrs, sizeof(*vcard->addrs) *
251 (vcard->num_addrs + 1));
254 for (i = off; i < len; i++)
256 VCARD_TYPETOKEN(vcard->addrs[vcard->num_addrs].type);
257 VCARD_TOKEN(vcard->addrs[vcard->num_addrs].pbox);
258 VCARD_TOKEN(vcard->addrs[vcard->num_addrs].ext_addr);
259 VCARD_TOKEN(vcard->addrs[vcard->num_addrs].street_addr);
260 VCARD_TOKEN(vcard->addrs[vcard->num_addrs].city);
261 VCARD_TOKEN(vcard->addrs[vcard->num_addrs].state);
262 VCARD_TOKEN(vcard->addrs[vcard->num_addrs].code);
264 VCARD_LASTTOKEN(vcard->addrs[vcard->num_addrs].country);
267 } else if (!strncasecmp(val, "TEL;", 4)) {
268 vcard->tels = silc_realloc(vcard->tels, sizeof(*vcard->tels) *
269 (vcard->num_tels + 1));
272 for (i = off; i < len; i++)
275 VCARD_TYPETOKEN(vcard->tels[vcard->num_tels].type);
278 VCARD_LASTTOKEN(vcard->tels[vcard->num_tels].tel);
281 } else if (!strncasecmp(val, "EMAIL;", 6)) {
282 vcard->emails = silc_realloc(vcard->emails, sizeof(*vcard->emails) *
283 (vcard->num_emails + 1));
286 for (i = off; i < len; i++)
289 VCARD_TYPETOKEN(vcard->emails[vcard->num_emails].type);
292 VCARD_LASTTOKEN(vcard->emails[vcard->num_emails].address);
295 } else if (!strncasecmp(val, "NOTE:", 5)) {
299 vcard->note = silc_memdup(val + 5, len - 5);
300 } else if (!strncasecmp(val, "REV:", 4)) {
304 vcard->rev = silc_memdup(val + 4, len - 4);
305 } else if (!strncasecmp(val, VCARD_FOOTER, strlen(VCARD_FOOTER))) {
310 val = strchr(val, '\n');
318 if (!has_begin || !has_end) {
319 silc_vcard_free(vcard);
326 /* Allocate vcard context */
328 SilcVCard silc_vcard_alloc(void)
330 SilcVCard vcard = silc_calloc(1, sizeof(*vcard));
333 vcard->dynamic = TRUE;
337 /* Free the vcard structure */
339 void silc_vcard_free(SilcVCard vcard)
343 silc_free(vcard->full_name);
344 silc_free(vcard->family_name);
345 silc_free(vcard->first_name);
346 silc_free(vcard->middle_names);
347 silc_free(vcard->prefix);
348 silc_free(vcard->suffix);
349 silc_free(vcard->nickname);
350 silc_free(vcard->bday);
351 silc_free(vcard->title);
352 silc_free(vcard->role);
353 silc_free(vcard->org_name);
354 silc_free(vcard->org_unit);
355 silc_free(vcard->categories);
356 silc_free(vcard->class);
357 silc_free(vcard->url);
358 silc_free(vcard->label);
359 for (i = 0; i < vcard->num_addrs; i++) {
360 silc_free(vcard->addrs[i].type);
361 silc_free(vcard->addrs[i].pbox);
362 silc_free(vcard->addrs[i].ext_addr);
363 silc_free(vcard->addrs[i].street_addr);
364 silc_free(vcard->addrs[i].city);
365 silc_free(vcard->addrs[i].state);
366 silc_free(vcard->addrs[i].code);
367 silc_free(vcard->addrs[i].country);
369 silc_free(vcard->addrs);
370 for (i = 0; i < vcard->num_tels; i++) {
371 silc_free(vcard->tels[i].type);
372 silc_free(vcard->tels[i].tel);
374 silc_free(vcard->tels);
375 for (i = 0; i < vcard->num_emails; i++) {
376 silc_free(vcard->emails[i].type);
377 silc_free(vcard->emails[i].address);
379 silc_free(vcard->emails);
380 silc_free(vcard->note);
381 silc_free(vcard->rev);
383 if (vcard->dynamic) {
384 memset(vcard, 'F', sizeof(*vcard));
389 /* Print card to file stream */
391 void silc_vcard_fprintf(SilcVCard vcard, FILE *stream)
394 fprintf(stream, "%s", VCARD_HEADER);
395 fprintf(stream, "%s", VCARD_VERSION);
396 if (vcard->full_name)
397 fprintf(stream, "FN:%s\n", vcard->full_name);
398 if (vcard->family_name)
399 fprintf(stream, "N:%s;%s;%s;%s;%s\n",
401 vcard->first_name ? vcard->first_name : "",
402 vcard->middle_names ? vcard->middle_names : "",
403 vcard->prefix ? vcard->prefix : "",
404 vcard->suffix ? vcard->suffix : "");
406 fprintf(stream, "NICKNAME:%s\n", vcard->nickname);
408 fprintf(stream, "BDAY:%s\n", vcard->bday);
410 fprintf(stream, "TITLE:%s\n", vcard->title);
412 fprintf(stream, "ROLE:%s\n", vcard->role);
414 fprintf(stream, "ORG:%s;%s\n", vcard->org_name,
415 vcard->org_unit ? vcard->org_unit : "");
416 if (vcard->categories)
417 fprintf(stream, "CATEGORIES:%s\n", vcard->categories);
419 fprintf(stream, "CLASS:%s\n", vcard->class);
421 fprintf(stream, "URL:%s\n", vcard->url);
423 fprintf(stream, "LABEL;%s\n", vcard->label);
424 for (i = 0; i < vcard->num_addrs; i++) {
425 fprintf(stream, "ADR;TYPE=%s:%s;%s;%s;%s;%s;%s;%s\n",
426 vcard->addrs[i].type,
427 vcard->addrs[i].pbox ? vcard->addrs[i].pbox : "",
428 vcard->addrs[i].ext_addr ? vcard->addrs[i].ext_addr : "",
429 vcard->addrs[i].street_addr ? vcard->addrs[i].street_addr : "",
430 vcard->addrs[i].city ? vcard->addrs[i].city : "",
431 vcard->addrs[i].state ? vcard->addrs[i].state : "",
432 vcard->addrs[i].code ? vcard->addrs[i].code : "",
433 vcard->addrs[i].country ? vcard->addrs[i].country : "");
435 for (i = 0; i < vcard->num_tels; i++) {
436 fprintf(stream, "TEL;TYPE=%s:%s\n",
438 vcard->tels[i].tel ? vcard->tels[i].tel : "");
440 for (i = 0; i < vcard->num_emails; i++) {
441 fprintf(stream, "EMAIL;TYPE=%s:%s\n",
442 vcard->emails[i].type,
443 vcard->emails[i].address ? vcard->emails[i].address : "");
446 fprintf(stream, "NOTE:%s\n", vcard->note);
448 fprintf(stream, "REV:%s\n", vcard->rev);
449 fprintf(stream, "%s", VCARD_FOOTER);