| 
									
										
										
										
											2019-06-21 12:35:08 -06:00
										 |  |  | package again | 
					
						
							| 
									
										
										
										
											2019-06-16 02:13:49 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2019-06-22 17:11:14 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	webhooks "git.rootprojects.org/root/go-again/webhooks" | 
					
						
							| 
									
										
										
										
											2019-06-16 02:13:49 -06:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-21 12:35:08 -06:00
										 |  |  | type Schedule struct { | 
					
						
							|  |  |  | 	NextRunAt time.Time | 
					
						
							| 
									
										
										
										
											2019-06-22 17:11:14 -06:00
										 |  |  | 	Webhooks  []webhooks.Webhook | 
					
						
							| 
									
										
										
										
											2019-06-21 12:35:08 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-16 02:13:49 -06:00
										 |  |  | // https://yourbasic.org/golang/time-change-convert-location-timezone/ | 
					
						
							|  |  |  | // https://sebest.github.io/post/create-a-small-docker-image-for-a-golang-binary/ | 
					
						
							|  |  |  | // https://github.com/FKSE/docker-golang-base | 
					
						
							|  |  |  | // tar cfz zoneinfo.tar.gz /usr/share/zoneinfo | 
					
						
							|  |  |  | // git clone https://github.com/eggert/tz | 
					
						
							|  |  |  | // grep '^Rule' -r tz/ | cut -f8-10 | egrep -iv 'Rule|SAVE' | 
					
						
							|  |  |  | // egrep '\s0:30' -r tz/ | 
					
						
							| 
									
										
										
										
											2019-06-21 12:35:08 -06:00
										 |  |  | func Run() { | 
					
						
							| 
									
										
										
										
											2019-06-16 02:13:49 -06:00
										 |  |  | 	// blacklist "", "Local" | 
					
						
							|  |  |  | 	// UTC to TZ should always be correct | 
					
						
							|  |  |  | 	// TZ to UTC may not be correct | 
					
						
							|  |  |  | 	now := time.Now() | 
					
						
							|  |  |  | 	fmt.Println("Now", now.Format(time.RFC3339)) | 
					
						
							|  |  |  | 	loc, err := time.LoadLocation("America/Phoenix") | 
					
						
							|  |  |  | 	if nil != err { | 
					
						
							|  |  |  | 		panic(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	fmt.Println("Loc", now.In(loc)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* boundary checks */ | 
					
						
							| 
									
										
										
										
											2019-06-18 16:24:58 -06:00
										 |  |  | 	for _, st := range [][]int{ | 
					
						
							|  |  |  | 		[]int{2019, 11, 10, 01, 02, 56, 0}, | 
					
						
							|  |  |  | 		[]int{2019, 11, 10, 01, 02, 60, 0}, | 
					
						
							|  |  |  | 		[]int{2019, 11, 10, 01, 60, 59, 0}, | 
					
						
							|  |  |  | 		[]int{2019, 11, 10, 24, 59, 59, 0}, | 
					
						
							|  |  |  | 		[]int{2019, 11, 10, 25, 59, 59, 0}, | 
					
						
							|  |  |  | 		[]int{2019, 11, 10, 23, 59, 59, 0}, | 
					
						
							|  |  |  | 		[]int{2019, 11, 31, 23, 0, 0, 0}, | 
					
						
							|  |  |  | 	} { | 
					
						
							|  |  |  | 		err := Exists(st, "America/Denver") | 
					
						
							|  |  |  | 		if nil != err { | 
					
						
							|  |  |  | 			fmt.Println(err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-06-16 02:13:49 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* funky times */ | 
					
						
							|  |  |  | 	tz := "America/Denver" | 
					
						
							|  |  |  | 	fmt.Println("funky") | 
					
						
							| 
									
										
										
										
											2019-06-19 09:38:57 -06:00
										 |  |  | 	for _, st := range [][]int{ | 
					
						
							|  |  |  | 		[]int{2019, 3, 10, 01, 59, 00, 0}, | 
					
						
							|  |  |  | 		[]int{2019, 3, 10, 02, 00, 00, 0}, | 
					
						
							|  |  |  | 		[]int{2019, 3, 10, 02, 01, 00, 0}, | 
					
						
							|  |  |  | 		[]int{2019, 3, 10, 02, 59, 00, 0}, | 
					
						
							|  |  |  | 		[]int{2019, 3, 10, 03, 00, 00, 0}, | 
					
						
							|  |  |  | 		[]int{2019, 3, 10, 03, 01, 00, 0}, | 
					
						
							|  |  |  | 		[]int{2019, 11, 03, 0, 59, 00, 0}, | 
					
						
							|  |  |  | 		[]int{2019, 11, 03, 01, 59, 00, 0}, | 
					
						
							|  |  |  | 		[]int{2019, 11, 03, 02, 00, 00, 0}, | 
					
						
							|  |  |  | 		[]int{2019, 11, 03, 02, 01, 00, 0}, | 
					
						
							|  |  |  | 		[]int{2019, 11, 03, 02, 59, 00, 0}, | 
					
						
							|  |  |  | 		[]int{2019, 11, 03, 03, 00, 00, 0}, | 
					
						
							|  |  |  | 		[]int{2019, 11, 03, 03, 01, 00, 0}, | 
					
						
							|  |  |  | 		[]int{2019, 11, 03, 04, 01, 00, 0}, | 
					
						
							|  |  |  | 		[]int{2019, 11, 03, 05, 01, 00, 0}, | 
					
						
							|  |  |  | 	} { | 
					
						
							|  |  |  | 		err := IsAmbiguous(st, tz) | 
					
						
							|  |  |  | 		if nil != err { | 
					
						
							|  |  |  | 			fmt.Println(err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-06-16 02:13:49 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-18 16:24:58 -06:00
										 |  |  | type ErrNoExist struct { | 
					
						
							|  |  |  | 	e string | 
					
						
							|  |  |  | 	t []int | 
					
						
							|  |  |  | 	z string | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (err ErrNoExist) Error() string { | 
					
						
							|  |  |  | 	return fmt.Sprintf("E_INVALID_TIME: '%#v' is not a valid timestamp at '%s': %s", err.t, err.z, err.e) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Check if the time is a real time in the given timezone. | 
					
						
							|  |  |  | // | 
					
						
							| 
									
										
										
										
											2019-06-19 09:38:57 -06:00
										 |  |  | // For example: 2:30am doesn't happen on 2019 March 10th according | 
					
						
							| 
									
										
										
										
											2019-06-18 16:24:58 -06:00
										 |  |  | // to America/Denver local time, due to the end of Daylight Savings Time, | 
					
						
							| 
									
										
										
										
											2019-06-19 09:38:57 -06:00
										 |  |  | // but it does happen in America/Phoenix. | 
					
						
							| 
									
										
										
										
											2019-06-18 16:24:58 -06:00
										 |  |  | // | 
					
						
							|  |  |  | // Also rejects times that are parsable and return a valid date object, | 
					
						
							|  |  |  | // but are not canonical, such as 24:60:75 November 31st, 2020. | 
					
						
							|  |  |  | // (which would be represented as `2020-12-02 02:01:15 +0000 UTC`) | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // Example of parsable, but non-canonical time: | 
					
						
							|  |  |  | // | 
					
						
							| 
									
										
										
										
											2019-06-19 09:38:57 -06:00
										 |  |  | //     var loc *time.Location | 
					
						
							|  |  |  | //     loc, _ = time.LoadLocation("America/Denver") | 
					
						
							|  |  |  | //     t := time.Date(2019, time.March, 10, 1, 30, 0, 0, loc) | 
					
						
							|  |  |  | //     fmt.Println(t, "==", t.UTC()) | 
					
						
							|  |  |  | //     // 2019-03-10 01:30:00 -0700 MST == 2019-03-10 08:30:00 +0000 UTC | 
					
						
							|  |  |  | //     t = t.Add(time.Duration(1) * time.Hour) | 
					
						
							|  |  |  | //     fmt.Println(t, "==", t.UTC()) | 
					
						
							|  |  |  | //     // 2019-03-10 03:30:00 -0600 MDT == 2019-03-10 09:30:00 +0000 UTC | 
					
						
							| 
									
										
										
										
											2019-06-18 16:24:58 -06:00
										 |  |  | // | 
					
						
							|  |  |  | // Example of a canonical, but non-parsable time (the 2016 leap second): | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | //     fmt.Println(time.Date(2016, time.December, 31, 23, 59, 60, 0, time.UTC)) | 
					
						
							|  |  |  | //     "2020-12-02 02:00:00 +0000 UTC" // should be "2016-12-31 23:59:60 +0000 UTC" | 
					
						
							| 
									
										
										
										
											2019-06-19 09:38:57 -06:00
										 |  |  | // | 
					
						
							| 
									
										
										
										
											2019-06-18 16:24:58 -06:00
										 |  |  | func Exists(st []int, tzstr string) error { | 
					
						
							| 
									
										
										
										
											2019-06-16 02:13:49 -06:00
										 |  |  | 	tz, err := time.LoadLocation(tzstr) | 
					
						
							|  |  |  | 	if nil != err { | 
					
						
							| 
									
										
										
										
											2019-06-18 16:24:58 -06:00
										 |  |  | 		return err | 
					
						
							| 
									
										
										
										
											2019-06-16 02:13:49 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-18 16:24:58 -06:00
										 |  |  | 	m := time.Month(st[1]) | 
					
						
							| 
									
										
										
										
											2019-06-16 02:13:49 -06:00
										 |  |  | 	t1 := time.Date(st[0], m, st[2], st[3], st[4], st[5], st[6], tz) | 
					
						
							|  |  |  | 	if st[5] != t1.Second() { | 
					
						
							| 
									
										
										
										
											2019-06-18 16:24:58 -06:00
										 |  |  | 		return ErrNoExist{ | 
					
						
							|  |  |  | 			t: st, | 
					
						
							|  |  |  | 			z: tzstr, | 
					
						
							|  |  |  | 			e: "invalid second, probably just bad math on your part", | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-06-16 02:13:49 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if st[4] != t1.Minute() { | 
					
						
							| 
									
										
										
										
											2019-06-18 16:24:58 -06:00
										 |  |  | 		return ErrNoExist{ | 
					
						
							|  |  |  | 			t: st, | 
					
						
							|  |  |  | 			z: tzstr, | 
					
						
							|  |  |  | 			e: "invalid minute, probably just bad math on your part, but perhaps a half-hour daylight savings or summer time", | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-06-16 02:13:49 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if st[3] != t1.Hour() { | 
					
						
							| 
									
										
										
										
											2019-06-18 16:24:58 -06:00
										 |  |  | 		return ErrNoExist{ | 
					
						
							|  |  |  | 			t: st, | 
					
						
							|  |  |  | 			z: tzstr, | 
					
						
							|  |  |  | 			e: "invalid hour, possibly a Daylight Savings or Summer Time error, or perhaps bad math on your part", | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-06-16 02:13:49 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if st[2] != t1.Day() { | 
					
						
							| 
									
										
										
										
											2019-06-18 16:24:58 -06:00
										 |  |  | 		return ErrNoExist{ | 
					
						
							|  |  |  | 			t: st, | 
					
						
							|  |  |  | 			z: tzstr, | 
					
						
							|  |  |  | 			e: "invalid day of month, most likely bad math on your part. Remember: 31 28¼ 31 30 31 30 31 31 30 31 30 31", | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-06-16 02:13:49 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if st[1] != int(t1.Month()) { | 
					
						
							| 
									
										
										
										
											2019-06-18 16:24:58 -06:00
										 |  |  | 		return ErrNoExist{ | 
					
						
							|  |  |  | 			t: st, | 
					
						
							|  |  |  | 			z: tzstr, | 
					
						
							|  |  |  | 			e: "invalid month, most likely bad math on your part. Remember: Decemberween isn't until next year", | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-06-16 02:13:49 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if st[0] != t1.Year() { | 
					
						
							| 
									
										
										
										
											2019-06-18 16:24:58 -06:00
										 |  |  | 		return ErrNoExist{ | 
					
						
							|  |  |  | 			t: st, | 
					
						
							|  |  |  | 			z: tzstr, | 
					
						
							|  |  |  | 			e: "invalid year, must have reached the end of time...", | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-06-16 02:13:49 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-18 16:24:58 -06:00
										 |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-19 09:38:57 -06:00
										 |  |  | // Check if the time happens more than once in a given timezone. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // For example: 1:30am happens only once on 2019 Nov 3rd according to | 
					
						
							|  |  |  | // America/Phoenix time but due to the start of Daylight Savings Time, | 
					
						
							|  |  |  | // it happens twice in America/Denver. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // Example of duplicate, non-canonical time: | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | //     var loc *time.Location | 
					
						
							|  |  |  | //     loc, _ = time.LoadLocation("America/Denver") | 
					
						
							|  |  |  | //     t = time.Date(2019, time.November, 3, 1, 30, 0, 0, loc) | 
					
						
							|  |  |  | //     fmt.Println(t, "==", t.UTC()) | 
					
						
							|  |  |  | //     // 2019-11-03 01:30:00 -0600 MDT == 2019-11-03 07:30:00 +0000 UTC | 
					
						
							|  |  |  | //     t = t.Add(time.Duration(1) * time.Hour) | 
					
						
							|  |  |  | //     fmt.Println(t, "==", t.UTC()) | 
					
						
							|  |  |  | //     // 2019-11-03 01:30:00 -0700 MST == 2019-11-03 08:30:00 +0000 UTC | 
					
						
							|  |  |  | //     t = t.Add(time.Duration(1) * time.Hour) | 
					
						
							|  |  |  | //     fmt.Println(t, "==", t.UTC()) | 
					
						
							|  |  |  | //     // 2019-11-03 02:30:00 -0700 MST == 2019-11-03 09:30:00 +0000 UTC | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | func IsAmbiguous(st []int, tzstr string) error { | 
					
						
							| 
									
										
										
										
											2019-06-16 02:13:49 -06:00
										 |  |  | 	// Does the time exist twice? | 
					
						
							|  |  |  | 	// (if I change the time in UTC, do I still get the same time) | 
					
						
							|  |  |  | 	// Note: Some timezones change by half or quarter hour | 
					
						
							|  |  |  | 	//       However, it seems that DST always changes by one or two whole hours | 
					
						
							|  |  |  | 	// Oh, and then there's Luthuania... | 
					
						
							|  |  |  | 	// Rule	LH	2008	max	-	Oct	Sun>=1	2:00	0:30	- | 
					
						
							|  |  |  | 	// | 
					
						
							|  |  |  | 	// https://en.wikipedia.org/wiki/Daylight_saving_time_by_country | 
					
						
							|  |  |  | 	// https://en.wikipedia.org/wiki/Winter_time_(clock_lag) | 
					
						
							|  |  |  | 	// https://en.wikipedia.org/wiki/Summer_time_in_Europe | 
					
						
							|  |  |  | 	// https://en.wikipedia.org/wiki/Daylight_saving_time_in_the_Americas | 
					
						
							|  |  |  | 	// If I change the time iadd or subtract time in UTC, do I see the same difference in TZ? | 
					
						
							| 
									
										
										
										
											2019-06-18 16:24:58 -06:00
										 |  |  | 	tz, err := time.LoadLocation(tzstr) | 
					
						
							|  |  |  | 	if nil != err { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	m := time.Month(st[1]) | 
					
						
							|  |  |  | 	t1 := time.Date(st[0], m, st[2], st[3], st[4], st[5], st[6], tz) | 
					
						
							| 
									
										
										
										
											2019-06-16 02:13:49 -06:00
										 |  |  | 	u1 := t1.UTC() | 
					
						
							|  |  |  | 	// A better way to do this would probably be to parse the timezone database, but... yeah... | 
					
						
							|  |  |  | 	for _, n := range []int{ /*-120, -60,*/ 30, 60, 120} { | 
					
						
							|  |  |  | 		t2 := time.Date(st[0], m, st[2], st[3], st[4]+n, st[5], st[6], tz) | 
					
						
							|  |  |  | 		u2 := t2.UTC() | 
					
						
							|  |  |  | 		if u1.Equal(u2) { | 
					
						
							|  |  |  | 			fmt.Println("Ambiguous Time") | 
					
						
							|  |  |  | 			fmt.Printf("%s, %s, %+d\n", t1, u1, n) | 
					
						
							|  |  |  | 			fmt.Printf("%s, %s, %+d\n", t2, u2, n) | 
					
						
							| 
									
										
										
										
											2019-06-19 09:38:57 -06:00
										 |  |  | 			return fmt.Errorf("Ambiguous") | 
					
						
							| 
									
										
										
										
											2019-06-16 02:13:49 -06:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	//ta := | 
					
						
							| 
									
										
										
										
											2019-06-18 16:24:58 -06:00
										 |  |  | 	return nil | 
					
						
							| 
									
										
										
										
											2019-06-16 02:13:49 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ////// | 
					
						
							|  |  |  | ////// 9:01 twice | 
					
						
							|  |  |  | ////// | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var d = new Date("3/10/2019, 01:59:00"); | 
					
						
							|  |  |  | console.log("tzUTC:" + tzUTC(d, 'America/Denver')); | 
					
						
							|  |  |  | // tzUTC:Sun, 10 Mar 2019 08:59:00 GMT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var d = new Date("3/10/2019, 02:01:00"); | 
					
						
							|  |  |  | console.log("tzUTC:" + tzUTC(d, 'America/Denver')); | 
					
						
							|  |  |  | // tzUTC:Sun, 10 Mar 2019 09:01:00 GMT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var d = new Date("3/10/2019, 02:59:00"); | 
					
						
							|  |  |  | console.log("tzUTC:" + tzUTC(d, 'America/Denver')); | 
					
						
							|  |  |  | // tzUTC:Sun, 10 Mar 2019 09:59:00 GMT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var d = new Date("3/10/2019, 03:01:00"); | 
					
						
							|  |  |  | console.log("tzUTC:" + tzUTC(d, 'America/Denver')); | 
					
						
							|  |  |  | // tzUTC:Sun, 10 Mar 2019 09:01:00 GMT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ////// | 
					
						
							|  |  |  | ////// 8:01 never | 
					
						
							|  |  |  | ////// | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var d = new Date("11/03/2019, 01:59:00"); | 
					
						
							|  |  |  | console.log("tzUTC:" + tzUTC(d, 'America/Denver')); | 
					
						
							|  |  |  | // tzUTC:Sun, 03 Nov 2019 07:59:00 GMT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var d = new Date("11/03/2019, 02:01:00"); | 
					
						
							|  |  |  | console.log("tzUTC:" + tzUTC(d, 'America/Denver')); | 
					
						
							|  |  |  | // tzUTC:Sun, 03 Nov 2019 09:01:00 GMT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var d = new Date("11/03/2019, 02:59:00"); | 
					
						
							|  |  |  | console.log("tzUTC:" + tzUTC(d, 'America/Denver')); | 
					
						
							|  |  |  | // tzUTC:Sun, 03 Nov 2019 09:59:00 GMT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var d = new Date("11/03/2019, 03:01:00"); | 
					
						
							|  |  |  | console.log("tzUTC:" + tzUTC(d, 'America/Denver')); | 
					
						
							|  |  |  | tzUTC:Sun, 03 Nov 2019 10:01:00 GMT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | */ |