Skip to content
Snippets Groups Projects
say.c 109 KiB
Newer Older
  • Learn to ignore specific revisions
  • 	char sndfile[256], nextmsg[256];
    
    	ast_localtime(&time,&tm,timezone);
    
    	for (offset=0 ; format[offset] != '\0' ; offset++) {
    		ast_log(LOG_DEBUG, "Parsing %c (offset %d) in %s\n", format[offset], offset, format);
    		switch (format[offset]) {
    			/* NOTE:  if you add more options here, please try to be consistent with strftime(3) */
    			case '\'':
    				/* Literal name of a sound file */
    				sndoffset=0;
    				for (sndoffset=0 ; (format[++offset] != '\'') && (sndoffset < 256) ; sndoffset++)
    					sndfile[sndoffset] = format[offset];
    				sndfile[sndoffset] = '\0';
    				snprintf(nextmsg,sizeof(nextmsg), AST_SOUNDS "/%s", sndfile);
    				res = wait_file(chan,ints,nextmsg,lang);
    				break;
    			case 'A':
    			case 'a':
    				/* Sunday - Saturday */
    				snprintf(nextmsg,sizeof(nextmsg), "digits/day-%d", tm.tm_wday);
    				res = wait_file(chan,ints,nextmsg,lang);
    				break;
    			case 'B':
    			case 'b':
    			case 'h':
    				/* January - December */
    				snprintf(nextmsg,sizeof(nextmsg), "digits/mon-%d", tm.tm_mon);
    				res = wait_file(chan,ints,nextmsg,lang);
    				break;
    			case 'd':
    			case 'e':
    				/* First - Thirtyfirst */
    				if (!(tm.tm_mday % 10) || (tm.tm_mday < 10)) {
    					snprintf(nextmsg,sizeof(nextmsg), "digits/h-%d", tm.tm_mday);
    					res = wait_file(chan,ints,nextmsg,lang);
    				} else {
    					snprintf(nextmsg,sizeof(nextmsg), "digits/h-%dh", tm.tm_mday - (tm.tm_mday % 10));
    					res = wait_file(chan,ints,nextmsg,lang);
    					if(!res) {
    						snprintf(nextmsg,sizeof(nextmsg), "digits/h-%d", tm.tm_mday % 10);
    						res = wait_file(chan,ints,nextmsg,lang);
    					}
    				}
    				break;
    			case 'Y':
    				/* Year */
    				if (tm.tm_year > 99) {
    					res = wait_file(chan,ints, "digits/2",lang);
    					if (!res) {
    						res = wait_file(chan,ints, "digits/thousand",lang);
    					}
    					if (tm.tm_year > 100) {
    						if (!res) {
    							snprintf(nextmsg,sizeof(nextmsg), "digits/%d", (tm.tm_year - 100) / 10);
    							res = wait_file(chan,ints,nextmsg,lang);
    							if (!res) {
    								snprintf(nextmsg,sizeof(nextmsg), "digits/%d", (tm.tm_year - 100) % 10);
    								res = wait_file(chan,ints,nextmsg,lang);
    							}
    						}
    					}
    					if (!res) {
    						res = wait_file(chan,ints, "digits/year",lang);
    					}
    				} else {
    					if (tm.tm_year < 1) {
    						/* I'm not going to handle 1900 and prior */
    						/* We'll just be silent on the year, instead of bombing out. */
    					} else {
    						res = wait_file(chan,ints, "digits/1",lang);
    						if (!res) {
    							res = wait_file(chan,ints, "digits/9",lang);
    						}
    						if (!res) {
    							if (tm.tm_year <= 9) {
    								/* 1901 - 1909 */
    								res = wait_file(chan,ints, "digits/0",lang);
    								if (!res) {
    									snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_year);
    									res = wait_file(chan,ints,nextmsg,lang);
    								}
    							} else {
    								/* 1910 - 1999 */
    								snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_year / 10);
    								res = wait_file(chan,ints,nextmsg,lang);
    								if (!res) {
    									snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_year % 10);
    									res = wait_file(chan,ints,nextmsg,lang);
    								}
    							}
    						}
    					}
    					if (!res) {
    						res = wait_file(chan,ints, "digits/year",lang);
    					}
    				}
    				break;
    			case 'I':
    			case 'l':
    				/* 12-Hour */
    				if (tm.tm_hour == 0)
    					snprintf(nextmsg,sizeof(nextmsg), "digits/12");
    				else if (tm.tm_hour > 12)
    					snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_hour - 12);
    				else
    					snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_hour);
    				res = wait_file(chan,ints,nextmsg,lang);
    				if (!res) {
    					res = wait_file(chan,ints, "digits/oclock",lang);
    				}
    				break;
    			case 'H':
    			case 'k':
    				/* 24-Hour */
    				if (!(tm.tm_hour % 10) || tm.tm_hour < 10) {
    					if (tm.tm_hour < 10) {
    						res = wait_file(chan, ints, "digits/0", lang);
    					}
    					snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_hour);
    					res = wait_file(chan,ints,nextmsg,lang);
    				} else {
    					snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_hour - (tm.tm_hour % 10));
    					res = wait_file(chan,ints,nextmsg,lang);
    					if (!res) {
    						snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_hour % 10);
    						res = wait_file(chan,ints,nextmsg,lang);
    					}
    				}
    				if (!res) {
    					res = wait_file(chan,ints, "digits/oclock",lang);
    				}
    				break;
    			case 'M':
    				/* Minute */
    				if (!(tm.tm_min % 10) || tm.tm_min < 10) {
    					if (tm.tm_min < 10) {
    						res = wait_file(chan, ints, "digits/0", lang);
    					}
    					snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_min);
    					res = wait_file(chan,ints,nextmsg,lang);
    				} else {
    					snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_min - (tm.tm_min % 10));
    					res = wait_file(chan,ints,nextmsg,lang);
    					if (!res) {
    						snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_min % 10);
    						res = wait_file(chan,ints,nextmsg,lang);
    					}
    				}
    				if (!res) {
    					res = wait_file(chan,ints, "digits/minute",lang);
    				}
    				break;
    			case 'P':
    			case 'p':
    				/* AM/PM */
    				if (tm.tm_hour > 11)
    					snprintf(nextmsg,sizeof(nextmsg), "digits/p-m");
    				else
    					snprintf(nextmsg,sizeof(nextmsg), "digits/a-m");
    				res = wait_file(chan,ints,nextmsg,lang);
    				break;
    			case 'Q':
    				/* Shorthand for "Today", "Yesterday", or ABdY */
    				{
    					struct timeval now;
    					struct tm tmnow;
    					time_t beg_today;
    
    					gettimeofday(&now,NULL);
    					ast_localtime(&now.tv_sec,&tmnow,timezone);
    					/* This might be slightly off, if we transcend a leap second, but never more off than 1 second */
    					/* In any case, it saves not having to do ast_mktime() */
    					beg_today = now.tv_sec - (tmnow.tm_hour * 3600) - (tmnow.tm_min * 60) - (tmnow.tm_sec);
    					if (beg_today < time) {
    						/* Today */
    						res = wait_file(chan,ints, "digits/today",lang);
    					} else if (beg_today - 86400 < time) {
    						/* Yesterday */
    						res = wait_file(chan,ints, "digits/yesterday",lang);
    					} else {
    						res = ast_say_date_with_format(chan, time, ints, lang, "YBdA", timezone);
    					}
    				}
    				break;
    			case 'q':
    				/* Shorthand for "" (today), "Yesterday", A (weekday), or ABdY */
    				{
    					struct timeval now;
    					struct tm tmnow;
    					time_t beg_today;
    
    					gettimeofday(&now,NULL);
    					ast_localtime(&now.tv_sec,&tmnow,timezone);
    					/* This might be slightly off, if we transcend a leap second, but never more off than 1 second */
    					/* In any case, it saves not having to do ast_mktime() */
    					beg_today = now.tv_sec - (tmnow.tm_hour * 3600) - (tmnow.tm_min * 60) - (tmnow.tm_sec);
    					if (beg_today < time) {
    						/* Today */
    					} else if ((beg_today - 86400) < time) {
    						/* Yesterday */
    						res = wait_file(chan,ints, "digits/yesterday",lang);
    					} else if (beg_today - 86400 * 6 < time) {
    						/* Within the last week */
    						res = ast_say_date_with_format(chan, time, ints, lang, "A", timezone);
    					} else {
    						res = ast_say_date_with_format(chan, time, ints, lang, "YBdA", timezone);
    					}
    				}
    				break;
    			case 'R':
    				res = ast_say_date_with_format(chan, time, ints, lang, "HM", timezone);
    				break;
    			case 'S':
    				/* Seconds */
    				if (!(tm.tm_sec % 10) || tm.tm_sec < 10) {
    					if (tm.tm_sec < 10) {
    						res = wait_file(chan, ints, "digits/0", lang);
    					}
    					snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_sec);
    					res = wait_file(chan,ints,nextmsg,lang);
    				} else {
    					snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_sec - (tm.tm_sec % 10));
    					res = wait_file(chan,ints,nextmsg,lang);
    					if (!res) {
    						snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_sec % 10);
    						res = wait_file(chan,ints,nextmsg,lang);
    					}
    				}
    				if (!res) {
    					res = wait_file(chan,ints, "digits/second",lang);
    				}
    				break;
    			case 'T':
    				res = ast_say_date_with_format(chan, time, ints, lang, "HMS", timezone);
    				break;
    			case ' ':
    			case '	':
    				/* Just ignore spaces and tabs */
    			break;
    			default:
    				/* Unknown character */
    				ast_log(LOG_WARNING, "Unknown character in datetime format %s: %c at pos %d\n", format, format[offset], offset);
    		}
    		/* Jump out on DTMF */
    		if (res) {
    			break;
    		}
    	}
    	return res;
    }
    
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_say_time(struct ast_channel *chan, time_t t, char *ints, char *lang)
    
    {
    	if (!strcasecmp(lang, "en") ) {	/* English syntax */
    		return(ast_say_time_en(chan, t, ints, lang));
    	} else if (!strcasecmp(lang, "nl") ) {	/* Dutch syntax */
    		return(ast_say_time_nl(chan, t, ints, lang));
    	} else if (!strcasecmp(lang, "pt") ) {	/* Portuguese syntax */
    		return(ast_say_time_pt(chan, t, ints, lang));
    
    	} else if (!strcasecmp(lang, "tw") ) {	/* Taiwanese syntax */
    		return(ast_say_time_tw(chan, t, ints, lang));
    
    	}
    
    	/* Default to English */
    	return(ast_say_time_en(chan, t, ints, lang));
    }
    
    /* English syntax */
    int ast_say_time_en(struct ast_channel *chan, time_t t, char *ints, char *lang)
    
    Mark Spencer's avatar
    Mark Spencer committed
    {
    
    Mark Spencer's avatar
    Mark Spencer committed
    	int res = 0;
    	int hour, pm=0;
    
    	localtime_r(&t,&tm);
    	hour = tm.tm_hour;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (!hour)
    		hour = 12;
    	else if (hour == 12)
    		pm = 1;
    	else if (hour > 12) {
    		hour -= 12;
    		pm = 1;
    	}
    	if (!res)
    
    		res = ast_say_number(chan, hour, ints, lang, (char *) NULL);
    
    	if (tm.tm_min > 9) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if (!res)
    
    			res = ast_say_number(chan, tm.tm_min, ints, lang, (char *) NULL);
    
    	} else if (tm.tm_min) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if (!res)
    
    			res = ast_streamfile(chan, "digits/oh", lang);
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if (!res)
    			res = ast_waitstream(chan, ints);
    		if (!res)
    
    			res = ast_say_number(chan, tm.tm_min, ints, lang, (char *) NULL);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	} else {
    		if (!res)
    
    			res = ast_streamfile(chan, "digits/oclock", lang);
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if (!res)
    			res = ast_waitstream(chan, ints);
    	}
    	if (pm) {
    		if (!res)
    			res = ast_streamfile(chan, "digits/p-m", lang);
    	} else {
    		if (!res)
    			res = ast_streamfile(chan, "digits/a-m", lang);
    	}
    	if (!res)
    		res = ast_waitstream(chan, ints);
    	return res;
    }
    
    
    /* Dutch syntax */
    int ast_say_time_nl(struct ast_channel *chan, time_t t, char *ints, char *lang)
    {
    	struct tm tm;
    	int res = 0;
    	int hour;
    	localtime_r(&t,&tm);
    	hour = tm.tm_hour;
    	if (!res)
    		res = ast_say_number(chan, hour, ints, lang, (char *) NULL);
    
    	if (!res)
    		res = ast_streamfile(chan, "digits/nl-uur", lang);
    	if (!res)
    		res = ast_waitstream(chan, ints);
    	if (!res)
    	    if (tm.tm_min > 0) 
    		res = ast_say_number(chan, tm.tm_min, ints, lang, NULL);
    	return res;
    }
    
    /* Portuguese syntax */
    int ast_say_time_pt(struct ast_channel *chan, time_t t, char *ints, char *lang)
    {
    	struct tm tm;
    	int res = 0;
    	int hour;
    	localtime_r(&t,&tm);
    	hour = tm.tm_hour;
    	if (!res)
    		res = ast_say_number(chan, hour, ints, lang, "f");
    	if (tm.tm_min) {
    		if (!res)
    			res = wait_file(chan, ints, "digits/pt-e", lang);
    		if (!res)
    			res = ast_say_number(chan, tm.tm_min, ints, lang, (char *) NULL);
    	} else {
    		if (!res)
    			res = wait_file(chan, ints, "digits/pt-hora", lang);
    		if (tm.tm_hour != 1)
    			if (!res)
    				res = wait_file(chan, ints, "digits/pt-sss", lang);
    	}
    	if (!res)
    		res = ast_say_number(chan, hour, ints, lang, (char *) NULL);
    	return res;
    }
    
    
    /* Taiwanese syntax */
    int ast_say_time_tw(struct ast_channel *chan, time_t t, char *ints, char *lang)
    {
    	struct tm tm;
    	int res = 0;
    	int hour, pm=0;
    	localtime_r(&t,&tm);
    	hour = tm.tm_hour;
    	if (!hour)
    		hour = 12;
    	else if (hour == 12)
    		pm = 1;
    	else if (hour > 12) {
    		hour -= 12;
    		pm = 1;
    	}
    	if (pm) {
    		if (!res)
    			res = ast_streamfile(chan, "digits/p-m", lang);
    	} else {
    		if (!res)
    			res = ast_streamfile(chan, "digits/a-m", lang);
    	}
    	if (!res)
    		res = ast_waitstream(chan, ints);
    	if (!res)
    		res = ast_say_number(chan, hour, ints, lang, (char *) NULL);
    	if (!res)
    		res = ast_streamfile(chan, "digits/oclock", lang);
    	if (!res)
    		res = ast_waitstream(chan, ints);
    	if (!res)
    		res = ast_say_number(chan, tm.tm_min, ints, lang, (char *) NULL);
    	if (!res)
    		res = ast_streamfile(chan, "digits/minute", lang);
    	if (!res)
    		res = ast_waitstream(chan, ints);
    	return res;
    }
    
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_say_datetime(struct ast_channel *chan, time_t t, char *ints, char *lang)
    
    {
    	if (!strcasecmp(lang, "en") ) {	/* English syntax */
    		return(ast_say_datetime_en(chan, t, ints, lang));
    	} else if (!strcasecmp(lang, "nl") ) {	/* Dutch syntax */
    		return(ast_say_datetime_nl(chan, t, ints, lang));
    	} else if (!strcasecmp(lang, "pt") ) {	/* Portuguese syntax */
    		return(ast_say_datetime_pt(chan, t, ints, lang));
    
    	} else if (!strcasecmp(lang, "tw") ) {	/* Taiwanese syntax */
    		return(ast_say_datetime_tw(chan, t, ints, lang));
    
    	}
    
    	/* Default to English */
    	return(ast_say_datetime_en(chan, t, ints, lang));
    }
    
    /* English syntax */
    int ast_say_datetime_en(struct ast_channel *chan, time_t t, char *ints, char *lang)
    {
    	struct tm tm;
    	char fn[256];
    	int res = 0;
    	int hour, pm=0;
    	localtime_r(&t,&tm);
    	if (!res) {
    		snprintf(fn, sizeof(fn), "digits/day-%d", tm.tm_wday);
    		res = ast_streamfile(chan, fn, lang);
    		if (!res)
    			res = ast_waitstream(chan, ints);
    	}
    	if (!res) {
    		snprintf(fn, sizeof(fn), "digits/mon-%d", tm.tm_mon);
    		res = ast_streamfile(chan, fn, lang);
    		if (!res)
    			res = ast_waitstream(chan, ints);
    	}
    	if (!res)
    		res = ast_say_number(chan, tm.tm_mday, ints, lang, (char *) NULL);
    
    	hour = tm.tm_hour;
    	if (!hour)
    		hour = 12;
    	else if (hour == 12)
    		pm = 1;
    	else if (hour > 12) {
    		hour -= 12;
    		pm = 1;
    	}
    	if (!res)
    		res = ast_say_number(chan, hour, ints, lang, (char *) NULL);
    
    	if (tm.tm_min > 9) {
    		if (!res)
    			res = ast_say_number(chan, tm.tm_min, ints, lang, (char *) NULL);
    	} else if (tm.tm_min) {
    		if (!res)
    			res = ast_streamfile(chan, "digits/oh", lang);
    		if (!res)
    			res = ast_waitstream(chan, ints);
    		if (!res)
    			res = ast_say_number(chan, tm.tm_min, ints, lang, (char *) NULL);
    	} else {
    		if (!res)
    			res = ast_streamfile(chan, "digits/oclock", lang);
    		if (!res)
    			res = ast_waitstream(chan, ints);
    	}
    	if (pm) {
    		if (!res)
    			res = ast_streamfile(chan, "digits/p-m", lang);
    	} else {
    		if (!res)
    			res = ast_streamfile(chan, "digits/a-m", lang);
    	}
    	if (!res)
    		res = ast_waitstream(chan, ints);
    	if (!res)
    		res = ast_say_number(chan, tm.tm_year + 1900, ints, lang, (char *) NULL);
    	return res;
    }
    
    /* Dutch syntax */
    int ast_say_datetime_nl(struct ast_channel *chan, time_t t, char *ints, char *lang)
    {
    	struct tm tm;
    	int res = 0;
    	localtime_r(&t,&tm);
    	res = ast_say_date(chan, t, ints, lang);
    	if (!res) {
    		res = ast_streamfile(chan, "digits/nl-om", lang);
    		if (!res)
    			res = ast_waitstream(chan, ints);
    	}
    	if (!res) 
    		ast_say_time(chan, t, ints, lang);
    	return res;
    }
    
    /* Portuguese syntax */
    int ast_say_datetime_pt(struct ast_channel *chan, time_t t, char *ints, char *lang)
    
    Mark Spencer's avatar
    Mark Spencer committed
    {
    
    Mark Spencer's avatar
    Mark Spencer committed
    	char fn[256];
    	int res = 0;
    	int hour, pm=0;
    
    	localtime_r(&t,&tm);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (!res) {
    
    		snprintf(fn, sizeof(fn), "digits/day-%d", tm.tm_wday);
    
    Mark Spencer's avatar
    Mark Spencer committed
    		res = ast_streamfile(chan, fn, lang);
    		if (!res)
    			res = ast_waitstream(chan, ints);
    	}
    	if (!res) {
    
    		snprintf(fn, sizeof(fn), "digits/mon-%d", tm.tm_mon);
    
    Mark Spencer's avatar
    Mark Spencer committed
    		res = ast_streamfile(chan, fn, lang);
    		if (!res)
    			res = ast_waitstream(chan, ints);
    	}
    	if (!res)
    
    		res = ast_say_number(chan, tm.tm_mday, ints, lang, (char *) NULL);
    
    	hour = tm.tm_hour;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (!hour)
    		hour = 12;
    	else if (hour == 12)
    		pm = 1;
    	else if (hour > 12) {
    		hour -= 12;
    		pm = 1;
    	}
    	if (!res)
    
    		res = ast_say_number(chan, hour, ints, lang, (char *) NULL);
    
    	if (tm.tm_min > 9) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if (!res)
    
    			res = ast_say_number(chan, tm.tm_min, ints, lang, (char *) NULL);
    
    	} else if (tm.tm_min) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if (!res)
    			res = ast_streamfile(chan, "digits/oh", lang);
    		if (!res)
    			res = ast_waitstream(chan, ints);
    		if (!res)
    
    			res = ast_say_number(chan, tm.tm_min, ints, lang, (char *) NULL);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	} else {
    		if (!res)
    			res = ast_streamfile(chan, "digits/oclock", lang);
    		if (!res)
    			res = ast_waitstream(chan, ints);
    	}
    	if (pm) {
    		if (!res)
    			res = ast_streamfile(chan, "digits/p-m", lang);
    	} else {
    		if (!res)
    			res = ast_streamfile(chan, "digits/a-m", lang);
    	}
    	if (!res)
    		res = ast_waitstream(chan, ints);
    	if (!res)
    
    		res = ast_say_number(chan, tm.tm_year + 1900, ints, lang, (char *) NULL);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return res;
    }
    
    
    /* Taiwanese syntax */
    int ast_say_datetime_tw(struct ast_channel *chan, time_t t, char *ints, char *lang)
    {
    	struct tm tm;
    	char fn[256];
    	int res = 0;
    	int hour, pm=0;
    	localtime_r(&t,&tm);
    	if (!res)
    		res = ast_say_number(chan, tm.tm_year + 1900, ints, lang, (char *) NULL);
    	if (!res) {
    		snprintf(fn, sizeof(fn), "digits/mon-%d", tm.tm_mon);
    		res = ast_streamfile(chan, fn, lang);
    		if (!res)
    			res = ast_waitstream(chan, ints);
    	}
    	if (!res)
    		res = ast_say_number(chan, tm.tm_mday, ints, lang, (char *) NULL);
    	if (!res) {
    		snprintf(fn, sizeof(fn), "digits/day-%d", tm.tm_wday);
    		res = ast_streamfile(chan, fn, lang);
     		if (!res)
    			res = ast_waitstream(chan, ints);
    	}
    
    	hour = tm.tm_hour;
    	if (!hour)
    		hour = 12;
    	else if (hour == 12)
    		pm = 1;
    	else if (hour > 12) {
    		hour -= 12;
    		pm = 1;
    	}
    	if (pm) {
    		if (!res)
    			res = ast_streamfile(chan, "digits/p-m", lang);
    	} else {
    		if (!res)
    			res = ast_streamfile(chan, "digits/a-m", lang);
    	}
    	if (!res)
    		res = ast_waitstream(chan, ints);
    	if (!res)
    		res = ast_say_number(chan, hour, ints, lang, (char *) NULL);
    	if (!res)
    		res = ast_streamfile(chan, "digits/oclock", lang);
    	if (!res)
    		res = ast_waitstream(chan, ints);
    	if (!res)
    		res = ast_say_number(chan, tm.tm_min, ints, lang, (char *) NULL);
    	if (!res)
    		res = ast_streamfile(chan, "digits/minute", lang);
    	if (!res)
    		res = ast_waitstream(chan, ints);
    	return res;
    }
    
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_say_datetime_from_now(struct ast_channel *chan, time_t t, char *ints, char *lang)
    
    {
    	if (!strcasecmp(lang, "en") ) {	/* English syntax */
    		return(ast_say_datetime_from_now_en(chan, t, ints, lang));
    	} else if (!strcasecmp(lang, "pt") ) {	/* Portuguese syntax */
    		return(ast_say_datetime_from_now_pt(chan, t, ints, lang));
    	}
    
    	/* Default to English */
    	return(ast_say_datetime_from_now_en(chan, t, ints, lang));
    }
    
    /* English syntax */
    int ast_say_datetime_from_now_en(struct ast_channel *chan, time_t t, char *ints, char *lang)
    
    Mark Spencer's avatar
    Mark Spencer committed
    {
    	int res=0;
    	time_t nowt;
    	int daydiff;
    
    	struct tm tm;
    	struct tm now;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	char fn[256];
    
    	time(&nowt);
    
    
    	localtime_r(&t,&tm);
    	localtime_r(&nowt,&now);
    	daydiff = now.tm_yday - tm.tm_yday;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if ((daydiff < 0) || (daydiff > 6)) {
    		/* Day of month and month */
    		if (!res) {
    
    			snprintf(fn, sizeof(fn), "digits/mon-%d", tm.tm_mon);
    
    Mark Spencer's avatar
    Mark Spencer committed
    			res = ast_streamfile(chan, fn, lang);
    			if (!res)
    				res = ast_waitstream(chan, ints);
    		}
    		if (!res)
    
    			res = ast_say_number(chan, tm.tm_mday, ints, lang, (char *) NULL);
    
    Mark Spencer's avatar
    Mark Spencer committed
    
    	} else if (daydiff) {
    		/* Just what day of the week */
    		if (!res) {
    
    			snprintf(fn, sizeof(fn), "digits/day-%d", tm.tm_wday);
    
    Mark Spencer's avatar
    Mark Spencer committed
    			res = ast_streamfile(chan, fn, lang);
    			if (!res)
    				res = ast_waitstream(chan, ints);
    		}
    	} /* Otherwise, it was today */
    	if (!res)
    		res = ast_say_time(chan, t, ints, lang);
    	return res;
    }
    
    
    /* Portuguese syntax */
    int ast_say_datetime_from_now_pt(struct ast_channel *chan, time_t t, char *ints, char *lang)
    {
    	int res=0;
    	time_t nowt;
    	int daydiff;
    	struct tm tm;
    	struct tm now;
    	char fn[256];
    
    	time(&nowt);
    
    	localtime_r(&t,&tm);
    	localtime_r(&nowt,&now);
    	daydiff = now.tm_yday - tm.tm_yday;
    	if ((daydiff < 0) || (daydiff > 6)) {
    		/* Day of month and month */
    		if (!res)
    			res = ast_say_number(chan, tm.tm_mday, ints, lang, (char *) NULL);
    		if (!res)
    			res = wait_file(chan, ints, "digits/pt-de", lang);
    		snprintf(fn, sizeof(fn), "digits/mon-%d", tm.tm_mon);
    		if (!res)
    			res = wait_file(chan, ints, fn, lang);
    	
    	} else if (daydiff) {
    		/* Just what day of the week */
    		snprintf(fn, sizeof(fn), "digits/day-%d", tm.tm_wday);
    		if (!res)
    			res = wait_file(chan, ints, fn, lang);
    	}	/* Otherwise, it was today */
    	snprintf(fn, sizeof(fn), "digits/pt-ah");
    	if (!res)
    		res = wait_file(chan, ints, fn, lang);
    	if (tm.tm_hour != 1)
    	if (!res)
    		res = wait_file(chan, ints, "digits/pt-sss", lang);
    	if (!res)
    		res = ast_say_time(chan, t, ints, lang);
    	return res;
    }