2 * edit.c edit string to ascii, and ascii to internal
4 * Kern Sibbald, December MMII
10 Copyright (C) 2000-2005 Kern Sibbald
12 This program is free software; you can redistribute it and/or
13 modify it under the terms of the GNU General Public License as
14 published by the Free Software Foundation; either version 2 of
15 the License, or (at your option) any later version.
17 This program is distributed in the hope that it will be useful,
18 but WITHOUT ANY WARRANTY; without even the implied warranty of
19 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20 General Public License for more details.
22 You should have received a copy of the GNU General Public
23 License along with this program; if not, write to the Free
24 Software Foundation, Inc., 59 Temple Place - Suite 330, Boston,
32 /* We assume ASCII input and don't worry about overflow */
33 uint64_t str_to_uint64(char *str)
35 register char *p = str;
36 register uint64_t value = 0;
41 while (B_ISSPACE(*p)) {
47 while (B_ISDIGIT(*p)) {
48 value = B_TIMES10(value) + *p - '0';
54 int64_t str_to_int64(char *str)
56 register char *p = str;
57 register int64_t value;
58 bool negative = false;
63 while (B_ISSPACE(*p)) {
68 } else if (*p == '-') {
72 value = str_to_uint64(p);
81 * Edit an integer number with commas, the supplied buffer
82 * must be at least 27 bytes long. The incoming number
83 * is always widened to 64 bits.
85 char *edit_uint64_with_commas(uint64_t val, char *buf)
88 * Replacement for sprintf(buf, "%" llu, val)
91 mbuf[sizeof(mbuf)-1] = 0;
92 int i = sizeof(mbuf)-2; /* edit backward */
97 mbuf[i--] = "0123456789"[val%10];
101 bstrncpy(buf, &mbuf[i+1], 27);
102 return add_commas(buf, buf);
106 * Edit an integer number, the supplied buffer
107 * must be at least 27 bytes long. The incoming number
108 * is always widened to 64 bits.
110 char *edit_uint64(uint64_t val, char *buf)
113 * Replacement for sprintf(buf, "%" llu, val)
116 mbuf[sizeof(mbuf)-1] = 0;
117 int i = sizeof(mbuf)-2; /* edit backward */
122 mbuf[i--] = "0123456789"[val%10];
126 bstrncpy(buf, &mbuf[i+1], 27);
130 char *edit_int64(int64_t val, char *buf)
133 * Replacement for sprintf(buf, "%" llu, val)
136 bool negative = false;
137 mbuf[sizeof(mbuf)-1] = 0;
138 int i = sizeof(mbuf)-2; /* edit backward */
147 mbuf[i--] = "0123456789"[val%10];
154 bstrncpy(buf, &mbuf[i+1], 27);
160 * Given a string "str", separate the numeric part into
161 * str, and the modifier into mod.
163 static bool get_modifier(char *str, char *num, int num_len, char *mod, int mod_len)
165 int i, len, num_begin, num_end, mod_begin, mod_end;
167 strip_trailing_junk(str);
170 for (i=0; i<len; i++) {
171 if (!B_ISSPACE(str[i])) {
177 /* Walk through integer part */
178 for ( ; i<len; i++) {
179 if (!B_ISDIGIT(str[i]) && str[i] != '.') {
184 if (num_len > (num_end - num_begin + 1)) {
185 num_len = num_end - num_begin + 1;
190 /* Eat any spaces in front of modifier */
191 for ( ; i<len; i++) {
192 if (!B_ISSPACE(str[i])) {
197 for ( ; i<len; i++) {
198 if (!B_ISALPHA(str[i])) {
203 if (mod_len > (mod_end - mod_begin + 1)) {
204 mod_len = mod_end - mod_begin + 1;
206 Dmsg5(900, "str=%s: num_beg=%d num_end=%d mod_beg=%d mod_end=%d\n",
207 str, num_begin, num_end, mod_begin, mod_end);
208 bstrncpy(num, &str[num_begin], num_len);
209 bstrncpy(mod, &str[mod_begin], mod_len);
210 if (!is_a_number(num)) {
213 bstrncpy(str, &str[mod_end], len);
214 Dmsg2(900, "num=%s mod=%s\n", num, mod);
220 * Convert a string duration to utime_t (64 bit seconds)
221 * Returns 0: if error
222 1: if OK, and value stored in value
224 int duration_to_utime(char *str, utime_t *value)
227 double val, total = 0.0;
231 * The "n" = mins and months appears before minutes so that m maps
232 * to months. These "kludges" make it compatible with pre 1.31
235 static const char *mod[] = {"n", "seconds", "months", "minutes",
236 "hours", "days", "weeks", "quarters", "years", NULL};
237 static const int32_t mult[] = {60, 1, 60*60*24*30, 60,
238 60*60, 60*60*24, 60*60*24*7, 60*60*24*91, 60*60*24*365};
241 if (!get_modifier(str, num_str, sizeof(num_str), mod_str, sizeof(mod_str))) {
244 /* Now find the multiplier corresponding to the modifier */
245 mod_len = strlen(mod_str);
247 i = 1; /* assume seconds */
249 for (i=0; mod[i]; i++) {
250 if (strncasecmp(mod_str, mod[i], mod_len) == 0) {
254 if (mod[i] == NULL) {
255 i = 1; /* no modifier, assume secs */
258 Dmsg2(900, "str=%s: mult=%d\n", num_str, mult[i]);
260 val = strtod(num_str, NULL);
261 if (errno != 0 || val < 0) {
264 total += val * mult[i];
266 *value = (utime_t)total;
271 * Edit a utime "duration" into ASCII
273 char *edit_utime(utime_t val, char *buf, int buf_len)
276 static const int32_t mult[] = {60*60*24*365, 60*60*24*30, 60*60*24, 60*60, 60};
277 static const char *mod[] = {"year", "month", "day", "hour", "min"};
282 for (i=0; i<5; i++) {
283 times = (uint32_t)(val / mult[i]);
285 val = val - (utime_t)times * mult[i];
286 bsnprintf(mybuf, sizeof(mybuf), "%d %s%s ", times, mod[i], times>1?"s":"");
287 bstrncat(buf, mybuf, buf_len);
290 if (val == 0 && strlen(buf) == 0) {
291 bstrncat(buf, "0 secs", buf_len);
292 } else if (val != 0) {
293 bsnprintf(mybuf, sizeof(mybuf), "%d sec%s", (uint32_t)val, val>1?"s":"");
294 bstrncat(buf, mybuf, buf_len);
300 * Convert a size in bytes to uint64_t
301 * Returns 0: if error
302 1: if OK, and value stored in value
304 int size_to_uint64(char *str, int str_len, uint64_t *value)
310 static const char *mod[] = {"*", "k", "kb", "m", "mb", "g", "gb", NULL}; /* first item * not used */
311 const int64_t mult[] = {1, /* byte */
313 1000, /* kb kilobyte */
314 1048576, /* megabyte */
315 1000000, /* mb megabyte */
316 1073741824, /* gigabyte */
317 1000000000}; /* gb gigabyte */
319 if (!get_modifier(str, num_str, sizeof(num_str), mod_str, sizeof(mod_str))) {
322 /* Now find the multiplier corresponding to the modifier */
323 mod_len = strlen(mod_str);
324 for (i=0; mod[i]; i++) {
325 if (strncasecmp(mod_str, mod[i], mod_len) == 0) {
329 if (mod[i] == NULL) {
330 i = 0; /* no modifier found, assume 1 */
332 Dmsg2(900, "str=%s: mult=%d\n", str, mult[i]);
334 val = strtod(num_str, NULL);
335 if (errno != 0 || val < 0) {
338 *value = (utime_t)(val * mult[i]);
343 * Check if specified string is a number or not.
344 * Taken from SQLite, cool, thanks.
346 bool is_a_number(const char *n)
348 bool digit_seen = false;
350 if( *n == '-' || *n == '+' ) {
353 while (B_ISDIGIT(*n)) {
357 if (digit_seen && *n == '.') {
359 while (B_ISDIGIT(*n)) { n++; }
361 if (digit_seen && (*n == 'e' || *n == 'E')
362 && (B_ISDIGIT(n[1]) || ((n[1]=='-' || n[1] == '+') && B_ISDIGIT(n[2])))) {
363 n += 2; /* skip e- or e+ or e digit */
364 while (B_ISDIGIT(*n)) { n++; }
366 return digit_seen && *n==0;
370 * Check if the specified string is an integer
372 bool is_an_integer(const char *n)
374 bool digit_seen = false;
375 while (B_ISDIGIT(*n)) {
379 return digit_seen && *n==0;
383 * Check if Bacula Resoure Name is valid
386 * Check if the Volume name has legal characters
387 * If ua is non-NULL send the message
389 bool is_name_valid(char *name, POOLMEM **msg)
393 /* Special characters to accept */
394 const char *accept = ":.-_ ";
396 /* Restrict the characters permitted in the Volume name */
397 for (p=name; *p; p++) {
398 if (B_ISALPHA(*p) || B_ISDIGIT(*p) || strchr(accept, (int)(*p))) {
402 Mmsg(msg, _("Illegal character \"%c\" in name.\n"), *p);
407 if (len >= MAX_NAME_LENGTH) {
409 Mmsg(msg, _("Name too long.\n"));
415 Mmsg(msg, _("Volume name must be at least one character long.\n"));
425 * Add commas to a string, which is presumably
428 char *add_commas(char *val, char *buf)
446 for (i=0; i < 3; i++) {
455 void d_msg(const char*, int, int, const char*, ...)
457 int main(int argc, char *argv[])
459 char *str[] = {"3", "3n", "3 hours", "3.5 day", "3 week", "3 m", "3 q", "3 years"};
464 for (int i=0; i<8; i++) {
466 if (!duration_to_utime(buf, &val)) {
467 printf("Error return from duration_to_utime for in=%s\n", str[i]);
470 edit_utime(val, outval);
471 printf("in=%s val=%" lld " outval=%s\n", str[i], val, outval);