File: namefixup.c

package info (click to toggle)
mailfromd 9.0-1
  • links: PTS, VCS
  • area: main
  • in suites: sid, trixie
  • size: 11,268 kB
  • sloc: ansic: 56,643; sh: 22,791; yacc: 4,130; lex: 1,428; makefile: 914; lisp: 488; awk: 393; perl: 319; sed: 25
file content (123 lines) | stat: -rw-r--r-- 2,463 bytes parent folder | download
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
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
/* This file is part of Mailfromd.
   Copyright (C) 2005-2024 Sergey Poznyakoff

   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; either version 3, or (at your option)
   any later version.

   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, see <http://www.gnu.org/licenses/>. */

#ifdef HAVE_CONFIG_H
# include <config.h>
#endif

#include <stdlib.h>
#include <mailutils/alloc.h>

#include "libmf.h"

struct namerec {
	struct namerec *next;
	char **nameptr;
};

static struct namerec *head, *tail;

void
mf_namefixup_register(char **ptr, const char *initval)
{
	struct namerec *nrec = mu_alloc(sizeof(*nrec));
	nrec->next = NULL;
	nrec->nameptr = ptr;
	*ptr = initval ? mu_strdup(initval) : NULL;
		
	if (tail)
		tail->next = nrec;
	else
		head = nrec;
	tail = nrec;
}

/*
 * Return length of the scheme prefix of name.
 */
static int
scheme_length(char const *name)
{
	int i;
	enum { sc_ini, sc_col, sc_slash } state = sc_ini;

	for (i = 0; name[i]; i++) {
		switch (state) {
		case sc_ini:
			if (name[i] == ':')
				state = sc_col;
			else if (!(mu_isalnum(name[i]) || name[i] == '_'))
				return 0;
			break;

		case sc_col:
			if (name[i] == '/')
				state = sc_slash;
			else
				return i;
			break;

		case sc_slash:
			if (name[i] == '/')
				return i + 1;
			else
				return i - 1;
		}
	}
	return state == sc_ini ? 0 : i;
}

void
mf_file_name_ptr_fixup(char **ptr, char *dir, size_t dirlen)
{
	if (*ptr) {
		char *name = *ptr;
		int slen = scheme_length(name);
		if (name[slen] != '/') {
			size_t olen = strlen(name);
			char *p = mu_realloc(name, dirlen + 1 + olen + 1);
			memmove(p + slen + dirlen + 1, p + slen, olen - slen + 1);
			memcpy(p + slen, dir, dirlen);
			p[slen+dirlen] = '/';
			*ptr = p;
		}
	}
}

void
mf_namefixup_run(char *dir)
{
	size_t dirlen = strlen(dir);
	struct namerec *p;

	for (p = head; p; p = p->next)
		mf_file_name_ptr_fixup(p->nameptr, dir, dirlen);
}

void
mf_namefixup_free()
{
	struct namerec *p = head;

	while (p) {
		struct namerec *next = p->next;
		free(p);
		p = next;
	}
}