Skip to content
Snippets Groups Projects
say.c 89.5 KiB
Newer Older
  • Learn to ignore specific revisions
  • 		For example 21 is not twenty-one, but there is a single word in 'it'.
    
    		Idem for 28 (ie when a the 2nd part of a compund number
    
    		There are exceptions also for hundred, thousand and million.
    
    		In english 100 = one hundred, 200 is two hundred.
    		In italian 100 = cento , like to say hundred (without one),
    		200 and more are like english.
    		
    
    		1000 is one thousand in en, 2000 is two thousand.
    		In it we have 1000 = mille , 2000 = 2 mila 
    
    		For million(s) we use the plural, if more than one
    		Also, one million is abbreviated in it, like on-million,
    		or 'un milione', not 'uno milione'.
    		So the right file is provided.
    		*/
    
    
    		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 == 21) {
    				snprintf(fn, sizeof(fn), "digits/%d", num);
    				num = 0;
    			} else if (num == 28) {
    				snprintf(fn, sizeof(fn), "digits/%d", num);
    				num = 0;
    			} else if (num == 31) {
    				snprintf(fn, sizeof(fn), "digits/%d", num);
    				num = 0;
    			} else if (num == 38) {
    				snprintf(fn, sizeof(fn), "digits/%d", num);
    				num = 0;
    			} else if (num == 41) {
    				snprintf(fn, sizeof(fn), "digits/%d", num);
    				num = 0;
    			} else if (num == 48) {
    				snprintf(fn, sizeof(fn), "digits/%d", num);
    				num = 0;
    			} else if (num == 51) {
    				snprintf(fn, sizeof(fn), "digits/%d", num);
    				num = 0;
    			} else if (num == 58) {
    				snprintf(fn, sizeof(fn), "digits/%d", num);
    				num = 0;
    			} else if (num == 61) {
    				snprintf(fn, sizeof(fn), "digits/%d", num);
    				num = 0;
    			} else if (num == 68) {
    				snprintf(fn, sizeof(fn), "digits/%d", num);
    				num = 0;
    			} else if (num == 71) {
    				snprintf(fn, sizeof(fn), "digits/%d", num);
    				num = 0;
    			} else if (num == 78) {
    				snprintf(fn, sizeof(fn), "digits/%d", num);
    				num = 0;
    			} else if (num == 81) {
    				snprintf(fn, sizeof(fn), "digits/%d", num);
    				num = 0;
    			} else if (num == 88) {
    				snprintf(fn, sizeof(fn), "digits/%d", num);
    				num = 0;
    			} else if (num == 91) {
    				snprintf(fn, sizeof(fn), "digits/%d", num);
    				num = 0;
    			} else if (num == 98) {
    				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) {
    					if ((num / 100) > 1) {
    						snprintf(fn, sizeof(fn), "digits/%d", (num/100));
    						playh++;
    					} else {
    						snprintf(fn, sizeof(fn), "digits/hundred");
    					}
    					num -= ((num / 100) * 100);
    				} else {
    					if (num < 1000000) { /* 1,000,000 */
    						if ((num/1000) > 1)
    							res = ast_say_number_full(chan, num / 1000, ints, language, audiofd, ctrlfd);
    						if (res)
    							return res;
    						tempnum = num;
    						num = num % 1000;
    						if ((tempnum / 1000) < 2)
    							snprintf(fn, sizeof(fn), "digits/thousand");
    						else /* for 1000 it says mille, for >1000 (eg 2000) says mila */
    							snprintf(fn, sizeof(fn), "digits/thousands");
    					} else {
    						if (num < 1000000000) { /* 1,000,000,000 */
    							if ((num / 1000000) > 1)
    								res = ast_say_number_full(chan, num / 1000000, ints, language, audiofd, ctrlfd);
    							if (res)
    								return res;
    							tempnum = num;
    							num = num % 1000000;
    							if ((tempnum / 1000000) < 2)
    								snprintf(fn, sizeof(fn), "digits/million");
    							else
    								snprintf(fn, sizeof(fn), "digits/millions");
    						} else {
    							ast_log(LOG_DEBUG, "Number '%d' is too big for me\n", num);
    							res = -1;
    						}
    					}
    				}
    			}
    			if (!res) {
    				if(!ast_streamfile(chan, fn, language)) {
    					if (audiofd && ctrlfd)
    						res = ast_waitstream_full(chan, ints, audiofd, ctrlfd);
    					else
    						res = ast_waitstream(chan, ints);
    				}
    				ast_stopstream(chan);
    			}
    		}
    	return res;
    
    /*--- ast_say_number_full_nl: dutch syntax */
    /* New files: digits/nl-en
    
    static int ast_say_number_full_nl(struct ast_channel *chan, int num, char *ints, char *language, int audiofd, int ctrlfd)
    
    	char fn[256] = "";
    	if (!num) 
    		return ast_say_digits_full(chan, 0,ints, language, audiofd, ctrlfd);
    
    	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) {
    			units = num % 10;
    			if (units > 0) {
    				res = ast_say_number_full_nl(chan, units, ints, language, audiofd, ctrlfd);
    				if (res)
    					return res;
    				num = num - units;
    				snprintf(fn, sizeof(fn), "digits/nl-en");
    			} else {
    				snprintf(fn, sizeof(fn), "digits/%d", num - units);
    				num = 0;
    			}
    		} else {
    			if (num < 1000) {
    				snprintf(fn, sizeof(fn), "digits/%d", (num/100));
    				playh++;
    				num -= ((num / 100) * 100);
    			} else {
    				if (num < 1000000) { /* 1,000,000 */
    					res = ast_say_number_full_en(chan, num / 1000, ints, language, audiofd, ctrlfd);
    					if (res)
    						return res;
    					num = num % 1000;
    					snprintf(fn, sizeof(fn), "digits/thousand");
    				} else {
    					if (num < 1000000000) { /* 1,000,000,000 */
    						res = ast_say_number_full_en(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) {
    			if(!ast_streamfile(chan, fn, language)) {
    				if (audiofd && ctrlfd)
    					res = ast_waitstream_full(chan, ints, audiofd, ctrlfd);
    				else
    					res = ast_waitstream(chan, ints);
    			}
    			ast_stopstream(chan);
    		}
    
    /* ast_say_number_full_pt: Portuguese syntax */
    /* 	Extra sounds needed: */
    /* 	For feminin all sound files end with F */
    /*	100E for 100+ something */
    /*	1000000S for plural */
    /*	pt-e for 'and' */
    static int ast_say_number_full_pt(struct ast_channel *chan, int num, char *ints, char *language, char *options, int audiofd, int ctrlfd)
    
    	int mf = 1;                            /* +1 = Masculin; -1 = Feminin */
    
    	if (!num) 
    		return ast_say_digits_full(chan, 0,ints, language, audiofd, ctrlfd);
    
    
    	if (options && !strncasecmp(options, "f",1))
    		mf = -1;
    
    	while(!res && num ) {
    		if (num < 20) {
    			if ((num == 1 || num == 2) && (mf < 0))
    				snprintf(fn, sizeof(fn), "digits/%dF", num);
    			else
    				snprintf(fn, sizeof(fn), "digits/%d", num);
    			num = 0;
    		} else if (num < 100) {
    			snprintf(fn, sizeof(fn), "digits/%d", (num / 10) * 10);
    			if (num % 10)
    				playh = 1;
    			num = num % 10;
    		} else if (num < 1000) {
    			if (num == 100)
    				snprintf(fn, sizeof(fn), "digits/100");
    			else if (num < 200)
    				snprintf(fn, sizeof(fn), "digits/100E");
    			else {
    				if (mf < 0 && num > 199)
    					snprintf(fn, sizeof(fn), "digits/%dF", (num / 100) * 100);
    				else
    					snprintf(fn, sizeof(fn), "digits/%d", (num / 100) * 100);
    				if (num % 100)
    					playh = 1;
    			}
    			num = num % 100;
    		} else if (num < 1000000) {
    			if (num > 1999) {
    				res = ast_say_number_full_pt(chan, (num / 1000) * mf, ints, language, options, audiofd, ctrlfd);
    				if (res)
    					return res;
    			}
    			snprintf(fn, sizeof(fn), "digits/1000");
    			if ((num % 1000) && ((num % 1000) < 100  || !(num % 100)))
    				playh = 1;
    			num = num % 1000;
    		} else if (num < 1000000000) {
    			res = ast_say_number_full_pt(chan, (num / 1000000), ints, language, options, audiofd, ctrlfd );
    			if (res)
    				return res;
    			if (num < 2000000)
    				snprintf(fn, sizeof(fn), "digits/1000000");
    			else
    				snprintf(fn, sizeof(fn), "digits/1000000S");
     
    			if ((num % 1000000) &&
    				// no thousands
    				((!((num / 1000) % 1000) && ((num % 1000) < 100 || !(num % 100))) ||
    				// no hundreds and below
    				(!(num % 1000) && (((num / 1000) % 1000) < 100 || !((num / 1000) % 100))) ) )
    				playh = 1;
    			num = num % 1000000;
    		}
    		if (!res && playh) {
    			res = wait_file(chan, ints, "digits/pt-e", language);
    			ast_stopstream(chan);
    			playh = 0;
    		}
    		if (!res) {
    			if(!ast_streamfile(chan, fn, language)) {
    				if (audiofd && ctrlfd)
    					res = ast_waitstream_full(chan, ints, audiofd, ctrlfd);			else
    					res = ast_waitstream(chan, ints);
    			}
    			ast_stopstream(chan);
    		}
    	}
    	return res;
    }
    
    /*--- ast_say_number_full_se: Swedish/Norwegian syntax */
    static int ast_say_number_full_se(struct ast_channel *chan, int num, char *ints, char *language, char *options, int audiofd, int ctrlfd)
    {
    	int res = 0;
    	int playh = 0;
    	char fn[256] = "";
    	int cn = 1;		/* +1 = Commune; -1 = Neutrum */
    	if (!num) 
    		return ast_say_digits_full(chan, 0,ints, language, audiofd, ctrlfd);
    	if (options && !strncasecmp(options, "n",1)) cn = -1;
    
    	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 == 1 && cn == -1) {	/* En eller ett? */
    			 	snprintf(fn, sizeof(fn), "digits/1N");
    				num = 0;
    			} else {
    				if (num < 1000){
    					snprintf(fn, sizeof(fn), "digits/%d", (num/100));
    					playh++;
    					num -= ((num / 100) * 100);
    				} else {
    					if (num < 1000000) { /* 1,000,000 */
    						res = ast_say_number_full_se(chan, num / 1000, ints, language, options, audiofd, ctrlfd);
    						if (res)
    							return res;
    						num = num % 1000;
    						snprintf(fn, sizeof(fn), "digits/thousand");
    					} else {
    						if (num < 1000000000) {	/* 1,000,000,000 */
    							res = ast_say_number_full_se(chan, num / 1000000, ints, language, options, 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) {
                                    if(!ast_streamfile(chan, fn, language)) {
                                        if (audiofd && ctrlfd)
                                            res = ast_waitstream_full(chan, ints, audiofd, ctrlfd);
                                        else
                                             res = ast_waitstream(chan, ints);
                                    }
                                    ast_stopstream(chan);
    
                            }
    			
    	}
    	return res;
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_say_date(struct ast_channel *chan, time_t t, char *ints, char *lang)
    
    {
    	if (!strcasecmp(lang,"en") ) {	/* English syntax */
    		return(ast_say_date_en(chan, t, ints, lang));
    	} else if (!strcasecmp(lang, "nl") ) {	/* Dutch syntax */
    		return(ast_say_date_nl(chan, t, ints, lang));
    	} else if (!strcasecmp(lang, "pt") ) {	/* Portuguese syntax */
    		return(ast_say_date_pt(chan, t, ints, lang));
    	}
    
    	/* Default to English */
    	return(ast_say_date_en(chan, t, ints, lang));
    }
    
    
    /* English syntax */
    int ast_say_date_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
    	char fn[256];
    	int res = 0;
    
    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);
    
    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, (char *) NULL);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return res;
    }
    
    
    /* Dutch syntax */
    int ast_say_date_nl(struct ast_channel *chan, time_t t, char *ints, char *lang)
    
    	struct tm tm;
    	char fn[256];
    	int res = 0;
    	ast_localtime(&t,&tm,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);
    	}
    	if (!res)
    		res = ast_say_number(chan, tm.tm_mday, 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_waitstream(chan, ints);
    
    	if (!res)
    		res = ast_say_number(chan, tm.tm_year + 1900, ints, lang, (char *) NULL);
    	return res;
    }
    
    /* Portuguese syntax */
    int ast_say_date_pt(struct ast_channel *chan, time_t t, char *ints, char *lang)
    {
    	struct tm tm;
    	char fn[256];
    	int res = 0;
    	ast_localtime(&t,&tm,NULL);
    	localtime_r(&t,&tm);
    	snprintf(fn, sizeof(fn), "digits/day-%d", tm.tm_wday);
    	if (!res)
    		res = wait_file(chan, ints, fn, lang);
    	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);
    	if (!res)
    		res = wait_file(chan, ints, "digits/pt-de", lang);
    	if (!res)
    		res = ast_say_number(chan, tm.tm_year + 1900, ints, lang, (char *) NULL);
    
    
    	return res;
    }
    
    int ast_say_date_with_format(struct ast_channel *chan, time_t time, char *ints, char *lang, char *format, char *timezone)
    
    {
    	if (!strcasecmp(lang, "en") ) {	/* English syntax */
    		return(ast_say_date_with_format_en(chan, time, ints, lang, format, timezone));
    
    	} else if (!strcasecmp(lang, "de") ) {	/* German syntax */
    		return(ast_say_date_with_format_de(chan, time, ints, lang, format, timezone));
    
    	} else if (!strcasecmp(lang, "es") || !strcasecmp(lang, "mx")) {	/* Spanish syntax */
    		return(ast_say_date_with_format_es(chan, time, ints, lang, format, timezone));
    
    	} else if (!strcasecmp(lang, "nl") ) {	/* Dutch syntax */
    		return(ast_say_date_with_format_nl(chan, time, ints, lang, format, timezone));
    	} else if (!strcasecmp(lang, "pt") ) {	/* Portuguese syntax */
    		return(ast_say_date_with_format_pt(chan, time, ints, lang, format, timezone));
    	}
    
    	/* Default to English */
    	return(ast_say_date_with_format_en(chan, time, ints, lang, format, timezone));
    }
    
    /* English syntax */
    int ast_say_date_with_format_en(struct ast_channel *chan, time_t time, char *ints, char *lang, char *format, char *timezone)
    
    {
    	struct tm tm;
    	int res=0, offset, sndoffset;
    	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 < 21) || (tm.tm_mday == 30)) {
    					snprintf(nextmsg,sizeof(nextmsg), "digits/h-%d", tm.tm_mday);
    					res = wait_file(chan,ints,nextmsg,lang);
    				} else if (tm.tm_mday == 31) {
    					/* "Thirty" and "first" */
    					res = wait_file(chan,ints, "digits/30",lang);
    					if (!res) {
    						res = wait_file(chan,ints, "digits/h-1",lang);
    					}
    				} else {
    					/* Between 21 and 29 - two sounds */
    					res = wait_file(chan,ints, "digits/20",lang);
    					if (!res) {
    						snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_mday - 20);
    						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) {
    							/* This works until the end of 2020 */
    							snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_year - 100);
    							res = wait_file(chan,ints,nextmsg,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/19",lang);
    						if (!res) {
    							if (tm.tm_year <= 9) {
    								/* 1901 - 1909 */
    								res = wait_file(chan,ints, "digits/oh",lang);
    								if (!res) {
    									snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_year);
    									res = wait_file(chan,ints,nextmsg,lang);
    								}
    							} else if (tm.tm_year <= 20) {
    								/* 1910 - 1920 */
    								snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_year);
    								res = wait_file(chan,ints,nextmsg,lang);
    							} else {
    								/* 1921 - 1999 */
    								int ten, one;
    								ten = tm.tm_year / 10;
    								one = tm.tm_year % 10;
    								snprintf(nextmsg,sizeof(nextmsg), "digits/%d", ten * 10);
    								res = wait_file(chan,ints,nextmsg,lang);
    								if (!res) {
    									if (one != 0) {
    										snprintf(nextmsg,sizeof(nextmsg), "digits/%d", one);
    										res = wait_file(chan,ints,nextmsg,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);
    				break;
    			case 'H':
    			case 'k':
    				/* 24-Hour */
    				if (format[offset] == 'H') {
    					/* e.g. oh-eight */
    					if (tm.tm_hour < 10) {
    						res = wait_file(chan,ints, "digits/oh",lang);
    					}
    				} else {
    					/* e.g. eight */
    					if (tm.tm_hour == 0) {
    						res = wait_file(chan,ints, "digits/oh",lang);
    					}
    				}
    				if (!res) {
    					if (tm.tm_hour != 0) {
    						int remainder = tm.tm_hour;
    						if (tm.tm_hour > 20) {
    							res = wait_file(chan,ints, "digits/20",lang);
    							remainder -= 20;
    						}
    						if (!res) {
    							snprintf(nextmsg,sizeof(nextmsg), "digits/%d", remainder);
    							res = wait_file(chan,ints,nextmsg,lang);
    						}
    					}
    				}
    				break;
    			case 'M':
    				/* Minute */
    				if (tm.tm_min == 0) {
    					res = wait_file(chan,ints, "digits/oclock",lang);
    				} else if (tm.tm_min < 10) {
    					res = wait_file(chan,ints, "digits/oh",lang);
    					if (!res) {
    						snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_min);
    						res = wait_file(chan,ints,nextmsg,lang);
    					}
    				} else if ((tm.tm_min < 21) || (tm.tm_min % 10 == 0)) {
    					snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_min);
    					res = wait_file(chan,ints,nextmsg,lang);
    				} else {
    					int ten, one;
    					ten = (tm.tm_min / 10) * 10;
    					one = (tm.tm_min % 10);
    					snprintf(nextmsg,sizeof(nextmsg), "digits/%d", ten);
    					res = wait_file(chan,ints,nextmsg,lang);
    					if (!res) {
    						/* Fifty, not fifty-zero */
    						if (one != 0) {
    							snprintf(nextmsg,sizeof(nextmsg), "digits/%d", one);
    							res = wait_file(chan,ints,nextmsg,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, "ABdY", 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, "ABdY", timezone);
    					}
    				}
    				break;
    			case 'R':
    				res = ast_say_date_with_format(chan, time, ints, lang, "HM", timezone);
    				break;
    			case 'S':
    				/* Seconds */
    				if (tm.tm_sec == 0) {
    					snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_sec);
    					res = wait_file(chan,ints,nextmsg,lang);
    				} else if (tm.tm_sec < 10) {
    					res = wait_file(chan,ints, "digits/oh",lang);
    					if (!res) {
    						snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_sec);
    						res = wait_file(chan,ints,nextmsg,lang);
    					}
    				} else if ((tm.tm_sec < 21) || (tm.tm_sec % 10 == 0)) {
    					snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_sec);
    					res = wait_file(chan,ints,nextmsg,lang);
    				} else {
    					int ten, one;
    					ten = (tm.tm_sec / 10) * 10;
    					one = (tm.tm_sec % 10);
    					snprintf(nextmsg,sizeof(nextmsg), "digits/%d", ten);
    					res = wait_file(chan,ints,nextmsg,lang);
    					if (!res) {
    						/* Fifty, not fifty-zero */
    						if (one != 0) {
    							snprintf(nextmsg,sizeof(nextmsg), "digits/%d", one);
    							res = wait_file(chan,ints,nextmsg,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;
    }
    
    /* German syntax */
    /* NB This currently is a 100% clone of the English syntax, just getting ready to make changes... */
    int ast_say_date_with_format_de(struct ast_channel *chan, time_t time, char *ints, char *lang, char *format, char *timezone)
    {
    	struct tm tm;
    	int res=0, offset, sndoffset;
    	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 */
    
    Mark Spencer's avatar
    Mark Spencer committed
    				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 */
    
    Mark Spencer's avatar
    Mark Spencer committed
    				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 < 21) || (tm.tm_mday == 30)) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    					snprintf(nextmsg,sizeof(nextmsg), "digits/h-%d", tm.tm_mday);
    
    					res = wait_file(chan,ints,nextmsg,lang);
    				} else if (tm.tm_mday == 31) {
    					/* "Thirty" and "first" */
    
    Mark Spencer's avatar
    Mark Spencer committed
    					res = wait_file(chan,ints, "digits/30",lang);
    
    Mark Spencer's avatar
    Mark Spencer committed
    						res = wait_file(chan,ints, "digits/h-1",lang);
    
    					}
    				} else {
    					/* Between 21 and 29 - two sounds */
    
    Mark Spencer's avatar
    Mark Spencer committed
    					res = wait_file(chan,ints, "digits/20",lang);
    
    Mark Spencer's avatar
    Mark Spencer committed
    						snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_mday - 20);
    
    						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) {
    							/* This works until the end of 2020 */
    							snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_year - 100);
    							res = wait_file(chan,ints,nextmsg,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/19",lang);
    						if (!res) {
    							if (tm.tm_year <= 9) {
    								/* 1901 - 1909 */
    								res = wait_file(chan,ints, "digits/oh",lang);
    								if (!res) {
    									snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_year);
    									res = wait_file(chan,ints,nextmsg,lang);
    								}
    							} else if (tm.tm_year <= 20) {
    								/* 1910 - 1920 */
    								snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_year);
    								res = wait_file(chan,ints,nextmsg,lang);
    							} else {
    								/* 1921 - 1999 */
    								int ten, one;
    								ten = tm.tm_year / 10;
    								one = tm.tm_year % 10;
    								snprintf(nextmsg,sizeof(nextmsg), "digits/%d", ten * 10);
    								res = wait_file(chan,ints,nextmsg,lang);
    								if (!res) {
    									if (one != 0) {
    										snprintf(nextmsg,sizeof(nextmsg), "digits/%d", one);
    										res = wait_file(chan,ints,nextmsg,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);
    				break;
    			case 'H':
    			case 'k':
    				/* 24-Hour */
    				if (format[offset] == 'H') {
    					/* e.g. oh-eight */
    					if (tm.tm_hour < 10) {
    						res = wait_file(chan,ints, "digits/oh",lang);
    					}
    				} else {
    					/* e.g. eight */
    					if (tm.tm_hour == 0) {
    						res = wait_file(chan,ints, "digits/oh",lang);
    					}
    				}
    				if (!res) {
    					if (tm.tm_hour != 0) {
    						int remainder = tm.tm_hour;
    						if (tm.tm_hour > 20) {
    							res = wait_file(chan,ints, "digits/20",lang);
    							remainder -= 20;
    						}
    						if (!res) {
    							snprintf(nextmsg,sizeof(nextmsg), "digits/%d", remainder);
    							res = wait_file(chan,ints,nextmsg,lang);
    						}
    					}
    				}
    				break;
    			case 'M':
    				/* Minute */
    				if (tm.tm_min == 0) {
    					res = wait_file(chan,ints, "digits/oclock",lang);
    				} else if (tm.tm_min < 10) {
    					res = wait_file(chan,ints, "digits/oh",lang);
    					if (!res) {
    						snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_min);
    						res = wait_file(chan,ints,nextmsg,lang);
    					}
    				} else if ((tm.tm_min < 21) || (tm.tm_min % 10 == 0)) {
    					snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_min);
    					res = wait_file(chan,ints,nextmsg,lang);
    				} else {
    					int ten, one;
    					ten = (tm.tm_min / 10) * 10;
    					one = (tm.tm_min % 10);
    					snprintf(nextmsg,sizeof(nextmsg), "digits/%d", ten);
    					res = wait_file(chan,ints,nextmsg,lang);
    					if (!res) {
    						/* Fifty, not fifty-zero */
    						if (one != 0) {
    							snprintf(nextmsg,sizeof(nextmsg), "digits/%d", one);
    							res = wait_file(chan,ints,nextmsg,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, "ABdY", 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, "ABdY", timezone);
    					}
    				}
    				break;
    			case 'R':
    				res = ast_say_date_with_format(chan, time, ints, lang, "HM", timezone);
    				break;
    			case 'S':
    				/* Seconds */
    				if (tm.tm_sec == 0) {
    					snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_sec);
    					res = wait_file(chan,ints,nextmsg,lang);
    				} else if (tm.tm_sec < 10) {
    					res = wait_file(chan,ints, "digits/oh",lang);
    					if (!res) {
    						snprintf(nextmsg,sizeof(nextmsg), "digits/%d", tm.tm_sec);
    						res = wait_file(chan,ints,nextmsg,lang);
    					}
    				} else if ((tm.tm_sec < 21) || (tm.tm_sec % 10 == 0)) {