*/
#include "bacula.h"
+#include <math.h>
/* We assume ASCII input and don't worry about overflow */
uint64_t str_to_uint64(char *str)
*/
int duration_to_utime(char *str, utime_t *value)
{
- int i, ch, len;
+ int i, len;
double val;
- static int mod[] = {'*', 's', 'n', 'h', 'd', 'w', 'm', 'q', 'y', 0};
- static int mult[] = {1, 1, 60, 60*60, 60*60*24, 60*60*24*7, 60*60*24*30,
- 60*60*24*91, 60*60*24*365};
+ /*
+ * The "n" = mins and months appears before minutes so that m maps
+ * to months. These "kludges" make it compatible with pre 1.31
+ * Baculas.
+ */
+ static const char *mod[] = {"n", "seconds", "months", "minutes",
+ "hours", "days", "weeks", "quarters", "years", NULL};
+ static const int32_t mult[] = {60, 1, 60*60*24*30, 60,
+ 60*60, 60*60*24, 60*60*24*7, 60*60*24*91, 60*60*24*365};
+ char mod_str[20];
+ int mod_len;
- /* Look for modifier */
+ /*
+ * Look for modifier by walking back looking for the first
+ * space or digit.
+ */
+ strip_trailing_junk(str);
len = strlen(str);
- ch = str[len - 1];
- i = 0;
- if (B_ISALPHA(ch)) {
- if (B_ISUPPER(ch)) {
- ch = tolower(ch);
+ /* Strip trailing spaces */
+ for (i=len; i>0; i--) {
+ if (!B_ISSPACE(str[i-1])) {
+ break;
}
- while (mod[++i] != 0) {
- if (ch == mod[i]) {
- len--;
- str[len] = 0; /* strip modifier */
- break;
- }
+ str[i-1] = 0;
+ }
+ /* Find beginning of the modifier */
+ for ( ; i>0; i--) {
+ if (!B_ISALPHA(str[i-1])) {
+ break;
}
}
- if (mod[i] == 0 || !is_a_number(str)) {
+ /* If not found, error */
+ if (i == 0 || i == len) {
+ Dmsg2(200, "error i=%d len=%d\n", i, len);
return 0;
}
+ /* Move modifier to mod_str */
+ bstrncpy(mod_str, &str[i], sizeof(mod_str));
+ mod_len = strlen(mod_str);
+ if (mod_len == 0) { /* Make sure we have a modifier */
+ Dmsg0(200, "No modifier found\n");
+ return 0;
+ }
+ Dmsg2(200, "in=%s mod=%s:\n", str, mod_str);
+ /* Backup over any spaces in front of modifier */
+ for ( ; i>0; i--) {
+ if (B_ISSPACE(str[i-1])) {
+ continue;
+ }
+ str[i] = 0;
+ break;
+ }
+ /* The remainder (beginning) should be our number */
+ if (!is_a_number(str)) {
+ Dmsg0(200, "input not a number\n");
+ return 0;
+ }
+ /* Now find the multiplier corresponding to the modifier */
+ for (i=0; mod[i]; i++) {
+ if (strncasecmp(mod_str, mod[i], mod_len) == 0) {
+ break;
+ }
+ }
+ if (mod[i] == NULL) {
+ Dmsg0(200, "Modifier not found\n");
+ return 0; /* modifer not found */
+ }
+ Dmsg2(200, "str=%s: mult=%d\n", str, mult[i]);
+ errno = 0;
val = strtod(str, NULL);
if (errno != 0 || val < 0) {
return 0;
}
- *value = (utime_t)(val * mult[i]);
+ *value = (utime_t)(val * mult[i]);
return 1;
}
1099511627776};/* terabyte */
#endif
- Dmsg0(400, "Enter sized to uint64\n");
+ Dmsg1(400, "Enter sized to uint64 str=%s\n", str);
/* Look for modifier */
ch = str[str_len - 1];
if (mod[i] == 0 || !is_a_number(str)) {
return 0;
}
- Dmsg3(400, "size str=:%s: %f i=%d\n", str, strtod(str, NULL), i);
+ Dmsg3(400, "size str=:%s: %lf i=%d\n", str, strtod(str, NULL), i);
- value = (uint64_t)strtod(str, NULL);
- Dmsg1(400, "Int value = %d\n", (int)value);
+ errno = 0;
+ value = strtod(str, NULL);
if (errno != 0 || value < 0) {
return 0;
}
+#if defined(HAVE_WIN32)
+ /* work around microsofts non handling of uint64 to double cvt*/
+ *rtn_value = (uint64_t)(value * (__int64)mult[i]);
+ Dmsg2(400, "Full value = %lf %" lld "\n", value * (__int64)mult[i],
+ (uint64_t)(value * (__int64)mult[i]));
+#else
*rtn_value = (uint64_t)(value * mult[i]);
- Dmsg2(400, "Full value = %f %" lld "\n", strtod(str, NULL) * mult[i],
- value *mult[i]);
+ Dmsg2(400, "Full value = %lf %" lld "\n", value * mult[i],
+ (uint64_t)(value * mult[i]));
+#endif
return 1;
}
*/
int is_a_number(const char *n)
{
- int digit_seen = 0;
+ bool digit_seen = false;
if( *n == '-' || *n == '+' ) {
n++;
}
while (B_ISDIGIT(*n)) {
- digit_seen = 1;
+ digit_seen = true;
n++;
}
if (digit_seen && *n == '.') {
return digit_seen && *n==0;
}
+/*
+ * Check if the specified string is an integer
+ */
+int is_an_integer(const char *n)
+{
+ bool digit_seen = false;
+ while (B_ISDIGIT(*n)) {
+ digit_seen = true;
+ n++;
+ }
+ return digit_seen && *n==0;
+}
+
+/*
+ * Check if Bacula Resoure Name is valid
+ */
+/*
+ * Check if the Volume name has legal characters
+ * If ua is non-NULL send the message
+ */
+bool is_name_valid(char *name, POOLMEM **msg)
+{
+ int len;
+ char *p;
+ /* Special characters to accept */
+ const char *accept = ":.-_ ";
+
+ /* Restrict the characters permitted in the Volume name */
+ for (p=name; *p; p++) {
+ if (B_ISALPHA(*p) || B_ISDIGIT(*p) || strchr(accept, (int)(*p))) {
+ continue;
+ }
+ if (msg) {
+ Mmsg(msg, _("Illegal character \"%c\" in name.\n"), *p);
+ }
+ return false;
+ }
+ len = strlen(name);
+ if (len >= MAX_NAME_LENGTH) {
+ if (msg) {
+ Mmsg(msg, _("Name too long.\n"));
+ }
+ return false;
+ }
+ if (len == 0) {
+ if (msg) {
+ Mmsg(msg, _("Volume name must be at least one character long.\n"));
+ }
+ return false;
+ }
+ return true;
+}
+
+
+
/*
* Add commas to a string, which is presumably
* a number.
}
return buf;
}
+
+#ifdef TEST_PROGRAM
+void d_msg(char*, int, int, char*, ...)
+{}
+int main(int argc, char *argv[])
+{
+ char *str[] = {"3", "3n", "3 hours", "3.5 day", "3 week", "3 m", "3 q", "3 years"};
+ utime_t val;
+ char buf[100];
+ char outval[100];
+
+ for (int i=0; i<8; i++) {
+ strcpy(buf, str[i]);
+ if (!duration_to_utime(buf, &val)) {
+ printf("Error return from duration_to_utime for in=%s\n", str[i]);
+ continue;
+ }
+ edit_utime(val, outval);
+ printf("in=%s val=%lld outval=%s\n", str[i], val, outval);
+ }
+}
+#endif