Skip to content
Snippets Groups Projects
say.c 9.01 KiB
Newer Older
  • Learn to ignore specific revisions
  • Mark Spencer's avatar
    Mark Spencer committed
    /*
     * Asterisk -- A telephony toolkit for Linux.
     *
     * Say numbers and dates (maybe words one day too)
     * 
    
    Mark Spencer's avatar
    Mark Spencer committed
     * Copyright (C) 1999, Mark Spencer
    
    Mark Spencer's avatar
    Mark Spencer committed
     *
     * Mark Spencer <markster@linux-support.net>
     *
     * This program is free software, distributed under the terms of
     * the GNU General Public License
     */
    
    
    #include <sys/types.h>
    
    Mark Spencer's avatar
    Mark Spencer committed
    #include <asterisk/file.h>
    #include <asterisk/channel.h>
    #include <asterisk/logger.h>
    #include <asterisk/say.h>
    #include <stdio.h>
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_say_digit_str(struct ast_channel *chan, char *fn2, char *ints, char *lang)
    
    Mark Spencer's avatar
    Mark Spencer committed
    {
    
    	/* XXX Merge with full version? XXX */
    
    Mark Spencer's avatar
    Mark Spencer committed
    	char fn[256] = "";
    	int num = 0;
    	int res = 0;
    	while(fn2[num] && !res) {
    		snprintf(fn, sizeof(fn), "digits/%c", fn2[num]);
    
    Mark Spencer's avatar
    Mark Spencer committed
    		res = ast_streamfile(chan, fn, lang);
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if (!res) 
    
    Mark Spencer's avatar
    Mark Spencer committed
    			res = ast_waitstream(chan, ints);
    
    Mark Spencer's avatar
    Mark Spencer committed
    		ast_stopstream(chan);
    		num++;
    	}
    	return res;
    }
    
    
    int ast_say_digit_str_full(struct ast_channel *chan, char *fn2, char *ints, char *lang, int audiofd, int ctrlfd)
    {
    	char fn[256] = "";
    	int num = 0;
    	int res = 0;
    	while(fn2[num] && !res) {
    		snprintf(fn, sizeof(fn), "digits/%c", fn2[num]);
    		res = ast_streamfile(chan, fn, lang);
    		if (!res) 
    			res = ast_waitstream_full(chan, ints, audiofd, ctrlfd);
    		ast_stopstream(chan);
    		num++;
    	}
    	return res;
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_say_digits(struct ast_channel *chan, int num, char *ints, char *lang)
    
    Mark Spencer's avatar
    Mark Spencer committed
    {
    
    	/* XXX Should I be merged with say_digits_full XXX */
    
    Mark Spencer's avatar
    Mark Spencer committed
    	char fn2[256];
    	snprintf(fn2, sizeof(fn2), "%d", num);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return ast_say_digit_str(chan, fn2, ints, lang);
    
    Mark Spencer's avatar
    Mark Spencer committed
    }
    
    
    int ast_say_digits_full(struct ast_channel *chan, int num, char *ints, char *lang, int audiofd, int ctrlfd)
    {
    	char fn2[256];
    	snprintf(fn2, sizeof(fn2), "%d", num);
    	return ast_say_digit_str_full(chan, fn2, ints, lang, audiofd, ctrlfd);
    }
    
    int ast_say_number_full(struct ast_channel *chan, int num, char *ints, char *language, int audiofd, int ctrlfd)
    {
    	int res = 0;
    	int playh = 0;
    	char fn[256] = "";
    	if (!num) 
    		return ast_say_digits_full(chan, 0,ints, language, audiofd, ctrlfd);
    	if (0) {
    	/* XXX Only works for english XXX */
    	} else {
    		/* Use english numbers */
    		language = "en";
    		while(!res && (num || playh)) {
    			if (playh) {
    				snprintf(fn, sizeof(fn), "digits/hundred");
    				playh = 0;
    			} else
    			if (num < 20) {
    				snprintf(fn, sizeof(fn), "digits/%d", num);
    				num = 0;
    			} else
    			if (num < 100) {
    				snprintf(fn, sizeof(fn), "digits/%d", (num /10) * 10);
    				num -= ((num / 10) * 10);
    			} else {
    				if (num < 1000){
    					snprintf(fn, sizeof(fn), "digits/%d", (num/100));
    					playh++;
    					num -= ((num / 100) * 100);
    				} else {
    					if (num < 1000000) {
    						res = ast_say_number_full(chan, num / 1000, ints, language, audiofd, ctrlfd);
    						if (res)
    							return res;
    						num = num % 1000;
    						snprintf(fn, sizeof(fn), "digits/thousand");
    					} else {
    						if (num < 1000000000) {
    							res = ast_say_number_full(chan, num / 1000000, ints, language, audiofd, ctrlfd);
    							if (res)
    								return res;
    							num = num % 1000000;
    							snprintf(fn, sizeof(fn), "digits/million");
    						} else {
    							ast_log(LOG_DEBUG, "Number '%d' is too big for me\n", num);
    							res = -1;
    						}
    					}
    				}
    			}
    			if (!res) {
    				res = ast_streamfile(chan, fn, language);
    				if (!res) 
    					res = ast_waitstream_full(chan, ints, audiofd, ctrlfd);
    				ast_stopstream(chan);
    			}
    			
    		}
    	}
    	return res;
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_say_number(struct ast_channel *chan, int num, char *ints, char *language)
    
    Mark Spencer's avatar
    Mark Spencer committed
    {
    
    	/* XXX Should I be merged with ast_say_number_full XXX */
    
    Mark Spencer's avatar
    Mark Spencer committed
    	int res = 0;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	int playh = 0;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	char fn[256] = "";
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (!num) 
    
    Mark Spencer's avatar
    Mark Spencer committed
    		return ast_say_digits(chan, 0,ints, language);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (0) {
    	/* XXX Only works for english XXX */
    	} else {
    		/* Use english numbers */
    		language = "en";
    
    Mark Spencer's avatar
    Mark Spencer committed
    		while(!res && (num || playh)) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    			if (playh) {
    				snprintf(fn, sizeof(fn), "digits/hundred");
    				playh = 0;
    			} else
    			if (num < 20) {
    				snprintf(fn, sizeof(fn), "digits/%d", num);
    				num = 0;
    			} else
    			if (num < 100) {
    				snprintf(fn, sizeof(fn), "digits/%d", (num /10) * 10);
    				num -= ((num / 10) * 10);
    
    Mark Spencer's avatar
    Mark Spencer committed
    			} else {
    
    Mark Spencer's avatar
    Mark Spencer committed
    				if (num < 1000){
    					snprintf(fn, sizeof(fn), "digits/%d", (num/100));
    					playh++;
    
    Mark Spencer's avatar
    Mark Spencer committed
    					num -= ((num / 100) * 100);
    
    Mark Spencer's avatar
    Mark Spencer committed
    				} else {
    
    Mark Spencer's avatar
    Mark Spencer committed
    					if (num < 1000000) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    						res = ast_say_number(chan, num / 1000, ints, language);
    						if (res)
    							return res;
    
    Mark Spencer's avatar
    Mark Spencer committed
    						num = num % 1000;
    						snprintf(fn, sizeof(fn), "digits/thousand");
    					} else {
    						if (num < 1000000000) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    							res = ast_say_number(chan, num / 1000000, ints, language);
    							if (res)
    								return res;
    
    Mark Spencer's avatar
    Mark Spencer committed
    							num = num % 1000000;
    							snprintf(fn, sizeof(fn), "digits/million");
    						} else {
    							ast_log(LOG_DEBUG, "Number '%d' is too big for me\n", num);
    							res = -1;
    						}
    					}
    
    Mark Spencer's avatar
    Mark Spencer committed
    				}
    
    Mark Spencer's avatar
    Mark Spencer committed
    			}
    
    Mark Spencer's avatar
    Mark Spencer committed
    			if (!res) {
    				res = ast_streamfile(chan, fn, language);
    				if (!res) 
    
    Mark Spencer's avatar
    Mark Spencer committed
    					res = ast_waitstream(chan, ints);
    
    Mark Spencer's avatar
    Mark Spencer committed
    				ast_stopstream(chan);
    			}
    			
    
    Mark Spencer's avatar
    Mark Spencer committed
    		}
    	}
    	return res;
    }
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_say_date(struct ast_channel *chan, time_t t, char *ints, char *lang)
    {
    
    Mark Spencer's avatar
    Mark Spencer committed
    	char fn[256];
    	int res = 0;
    
    	localtime_r(&t,&tm);
    	if (!&tm) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		ast_log(LOG_WARNING, "Unable to derive local time\n");
    		return -1;
    	}
    	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);
    
    Mark Spencer's avatar
    Mark Spencer committed
    
    	if (!res)
    		res = ast_waitstream(chan, ints);
    	if (!res)
    
    		res = ast_say_number(chan, tm.tm_year + 1900, ints, lang);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return res;
    }
    
    int ast_say_time(struct ast_channel *chan, time_t t, char *ints, char *lang)
    {
    
    Mark Spencer's avatar
    Mark Spencer committed
    	int res = 0;
    	int hour, pm=0;
    
    	localtime_r(&t,&tm);
    	if (!&tm) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		ast_log(LOG_WARNING, "Unable to derive local time\n");
    		return -1;
    	}
    
    	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);
    
    
    	if (tm.tm_min > 9) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if (!res)
    
    			res = ast_say_number(chan, tm.tm_min, ints, lang);
    	} 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);
    
    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);
    	return res;
    }
    
    int ast_say_datetime(struct ast_channel *chan, time_t t, char *ints, char *lang)
    {
    
    Mark Spencer's avatar
    Mark Spencer committed
    	char fn[256];
    	int res = 0;
    	int hour, pm=0;
    
    	localtime_r(&t,&tm);
    	if (!&tm) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		ast_log(LOG_WARNING, "Unable to derive local time\n");
    		return -1;
    	}
    	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);
    
    	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);
    
    
    	if (tm.tm_min > 9) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if (!res)
    
    			res = ast_say_number(chan, tm.tm_min, ints, lang);
    	} 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);
    
    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);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return res;
    }
    
    int ast_say_datetime_from_now(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;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	char fn[256];
    
    	time(&nowt);
    
    
    	localtime_r(&t,&tm);
    	if (!&tm) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		ast_log(LOG_WARNING, "Unable to derive local time\n");
    		return -1;
    	}
    
    	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);
    
    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;
    }