Compare commits

..

82 Commits

Author SHA1 Message Date
17cdae7bfb chore: Release
Some checks failed
Continuous integration / Check (push) Successful in 1m45s
Continuous integration / Test Suite (push) Successful in 2m35s
Continuous integration / Trunk (push) Successful in 2m12s
Continuous integration / Rustfmt (push) Failing after 42s
Continuous integration / Disallow unused dependencies (push) Successful in 2m36s
Continuous integration / build (push) Successful in 4m47s
2026-01-20 11:12:26 -08:00
f89135fce5 server: extract time range in addition to date from more calendar types 2026-01-20 11:12:16 -08:00
38c1d140bd Merge pull request 'chore(deps): update all non-major dependencies to v7.2.1' (#234) from renovate/all-minor-patch into master
Some checks failed
Continuous integration / Check (push) Has been cancelled
Continuous integration / Test Suite (push) Has been cancelled
Continuous integration / Trunk (push) Has been cancelled
Continuous integration / Rustfmt (push) Has been cancelled
Continuous integration / build (push) Has been cancelled
Continuous integration / Disallow unused dependencies (push) Has been cancelled
2026-01-20 10:47:32 -08:00
197ea049b2 chore(deps): update all non-major dependencies to v7.2.1
Some checks failed
Continuous integration / Check (push) Successful in 1m43s
Continuous integration / Test Suite (push) Successful in 2m59s
Continuous integration / Trunk (push) Successful in 1m4s
Continuous integration / Rustfmt (push) Failing after 1m34s
Continuous integration / build (push) Successful in 4m24s
Continuous integration / Disallow unused dependencies (push) Successful in 5m44s
2026-01-20 18:47:22 +00:00
f843166147 chore: Release
Some checks failed
Continuous integration / Check (push) Successful in 1m44s
Continuous integration / Test Suite (push) Successful in 2m49s
Continuous integration / Trunk (push) Successful in 1m53s
Continuous integration / Rustfmt (push) Failing after 1m35s
Continuous integration / build (push) Successful in 2m57s
Continuous integration / Disallow unused dependencies (push) Successful in 5m39s
2026-01-20 09:56:53 -08:00
943dd5c142 server: add Updated invitation parsing support 2026-01-20 09:56:16 -08:00
a1cf16350b server: big improvements for parsing all day events 2026-01-20 09:39:40 -08:00
626eca5619 Merge pull request 'chore(deps): update all non-major dependencies' (#233) from renovate/all-minor-patch into master
All checks were successful
Continuous integration / Check (push) Successful in 2m55s
Continuous integration / Test Suite (push) Successful in 4m20s
Continuous integration / Rustfmt (push) Successful in 42s
Continuous integration / Trunk (push) Successful in 2m11s
Continuous integration / build (push) Successful in 2m40s
Continuous integration / Disallow unused dependencies (push) Successful in 5m38s
2026-01-20 07:46:17 -08:00
cb77f83607 chore(deps): update all non-major dependencies
All checks were successful
Continuous integration / Check (push) Successful in 3m40s
Continuous integration / Test Suite (push) Successful in 6m30s
Continuous integration / Trunk (push) Successful in 21m18s
Continuous integration / Rustfmt (push) Successful in 1m38s
Continuous integration / build (push) Successful in 6m31s
Continuous integration / Disallow unused dependencies (push) Successful in 5m42s
2026-01-20 14:46:29 +00:00
b2f1431664 Merge pull request 'chore(deps): update rust crate html2text to v0.16.6' (#232) from renovate/all-minor-patch into master
All checks were successful
Continuous integration / Check (push) Successful in 2m54s
Continuous integration / Test Suite (push) Successful in 3m6s
Continuous integration / Rustfmt (push) Successful in 45s
Continuous integration / build (push) Successful in 2m28s
Continuous integration / Disallow unused dependencies (push) Successful in 2m27s
Continuous integration / Trunk (push) Successful in 21m7s
2026-01-18 02:31:16 -08:00
c541f4bd00 chore(deps): update rust crate html2text to v0.16.6
All checks were successful
Continuous integration / Check (push) Successful in 1m45s
Continuous integration / Test Suite (push) Successful in 6m44s
Continuous integration / Rustfmt (push) Successful in 1m37s
Continuous integration / Trunk (push) Successful in 7m38s
Continuous integration / Disallow unused dependencies (push) Successful in 2m31s
Continuous integration / build (push) Successful in 4m50s
2026-01-18 10:16:22 +00:00
4a20b1d4ba Merge pull request 'chore(deps): update rust crate human_format to v1.2.1' (#231) from renovate/all-minor-patch into master
All checks were successful
Continuous integration / Test Suite (push) Successful in 2m38s
Continuous integration / Check (push) Successful in 2m52s
Continuous integration / Rustfmt (push) Successful in 1m36s
Continuous integration / Trunk (push) Successful in 7m59s
Continuous integration / build (push) Successful in 7m12s
Continuous integration / Disallow unused dependencies (push) Successful in 2m35s
2026-01-16 20:46:17 -08:00
d54c90dc7b chore(deps): update rust crate human_format to v1.2.1
All checks were successful
Continuous integration / Check (push) Successful in 2m53s
Continuous integration / Test Suite (push) Successful in 2m58s
Continuous integration / Rustfmt (push) Successful in 59s
Continuous integration / build (push) Successful in 2m28s
Continuous integration / Disallow unused dependencies (push) Successful in 2m26s
Continuous integration / Trunk (push) Successful in 20m57s
2026-01-17 04:16:21 +00:00
f8a4a5d5b4 Merge pull request 'chore(deps): update all non-major dependencies to v7.2.0' (#230) from renovate/all-minor-patch into master
All checks were successful
Continuous integration / Check (push) Successful in 3m13s
Continuous integration / Test Suite (push) Successful in 2m20s
Continuous integration / Rustfmt (push) Successful in 1m7s
Continuous integration / Trunk (push) Successful in 2m11s
Continuous integration / build (push) Successful in 2m50s
Continuous integration / Disallow unused dependencies (push) Successful in 5m42s
2026-01-16 14:46:29 -08:00
8c99bb3ba1 chore(deps): update all non-major dependencies to v7.2.0
All checks were successful
Continuous integration / Check (push) Successful in 1m44s
Continuous integration / Test Suite (push) Successful in 3m28s
Continuous integration / Rustfmt (push) Successful in 49s
Continuous integration / build (push) Successful in 2m57s
Continuous integration / Disallow unused dependencies (push) Successful in 3m42s
Continuous integration / Trunk (push) Successful in 20m56s
2026-01-16 22:02:28 +00:00
c1a66fc548 Merge pull request 'fix(deps): update rust crate graphql_client to 0.16.0' (#229) from renovate/all-minor-patch into master
All checks were successful
Continuous integration / Check (push) Successful in 1m35s
Continuous integration / Test Suite (push) Successful in 2m59s
Continuous integration / Trunk (push) Successful in 1m23s
Continuous integration / Rustfmt (push) Successful in 44s
Continuous integration / build (push) Successful in 2m31s
Continuous integration / Disallow unused dependencies (push) Successful in 3m18s
2026-01-15 11:01:37 -08:00
f45bf002ba fix(deps): update rust crate graphql_client to 0.16.0
All checks were successful
Continuous integration / Test Suite (push) Successful in 2m36s
Continuous integration / Check (push) Successful in 3m29s
Continuous integration / Trunk (push) Successful in 1m12s
Continuous integration / Rustfmt (push) Successful in 1m35s
Continuous integration / build (push) Successful in 2m34s
Continuous integration / Disallow unused dependencies (push) Successful in 5m37s
2026-01-15 18:46:24 +00:00
630e9b68b7 Merge pull request 'chore(deps): update rust crate chrono to v0.4.43' (#228) from renovate/all-minor-patch into master
All checks were successful
Continuous integration / Check (push) Successful in 3m16s
Continuous integration / Test Suite (push) Successful in 2m30s
Continuous integration / Rustfmt (push) Successful in 57s
Continuous integration / Trunk (push) Successful in 2m8s
Continuous integration / build (push) Successful in 2m22s
Continuous integration / Disallow unused dependencies (push) Successful in 5m47s
2026-01-14 15:02:57 -08:00
787e10a989 chore(deps): update rust crate chrono to v0.4.43
All checks were successful
Continuous integration / Check (push) Successful in 1m43s
Continuous integration / Test Suite (push) Successful in 3m35s
Continuous integration / Trunk (push) Successful in 1m20s
Continuous integration / Rustfmt (push) Successful in 1m34s
Continuous integration / build (push) Successful in 3m21s
Continuous integration / Disallow unused dependencies (push) Successful in 5m51s
2026-01-14 21:47:44 +00:00
7de6a5f0fc chore: Release
All checks were successful
Continuous integration / Check (push) Successful in 2m12s
Continuous integration / Test Suite (push) Successful in 2m25s
Continuous integration / Trunk (push) Successful in 2m10s
Continuous integration / Rustfmt (push) Successful in 49s
Continuous integration / Disallow unused dependencies (push) Successful in 2m39s
Continuous integration / build (push) Successful in 4m35s
2026-01-14 13:37:20 -08:00
83230ba962 Merge branch 'renovate/lock-file-maintenance'
Some checks failed
Continuous integration / Test Suite (push) Successful in 3m47s
Continuous integration / Check (push) Successful in 5m46s
Continuous integration / Trunk (push) Successful in 2m12s
Continuous integration / Rustfmt (push) Successful in 1m38s
Continuous integration / build (push) Successful in 4m32s
Continuous integration / Disallow unused dependencies (push) Has been cancelled
2026-01-14 13:27:00 -08:00
6ef786c4e7 Manually update deps to a buildable rev 2026-01-14 13:26:30 -08:00
d4c29d7b98 chore(deps): lock file maintenance
Some checks failed
Continuous integration / Check (push) Failing after 1m34s
Continuous integration / Test Suite (push) Failing after 3m56s
Continuous integration / Rustfmt (push) Successful in 45s
Continuous integration / build (push) Failing after 3m11s
Continuous integration / Disallow unused dependencies (push) Failing after 3m7s
Continuous integration / Trunk (push) Successful in 21m26s
2026-01-14 18:31:51 +00:00
95a4f2e630 Merge pull request 'chore(deps): update all non-major dependencies' (#227) from renovate/all-minor-patch into master
All checks were successful
Continuous integration / Check (push) Successful in 1m38s
Continuous integration / Trunk (push) Successful in 1m53s
Continuous integration / Rustfmt (push) Successful in 53s
Continuous integration / Test Suite (push) Successful in 4m26s
Continuous integration / build (push) Successful in 2m20s
Continuous integration / Disallow unused dependencies (push) Successful in 5m42s
2026-01-14 10:31:19 -08:00
faa35dd65a chore(deps): update all non-major dependencies
All checks were successful
Continuous integration / Test Suite (push) Successful in 2m36s
Continuous integration / Check (push) Successful in 2m44s
Continuous integration / Trunk (push) Successful in 1m27s
Continuous integration / Rustfmt (push) Successful in 1m36s
Continuous integration / build (push) Successful in 2m48s
Continuous integration / Disallow unused dependencies (push) Successful in 5m37s
2026-01-14 13:16:29 +00:00
7c54c6b9c3 Merge pull request 'fix(deps): update rust crate quick-xml to 0.39.0' (#225) from renovate/all-minor-patch into master
All checks were successful
Continuous integration / Test Suite (push) Successful in 2m29s
Continuous integration / Check (push) Successful in 2m38s
Continuous integration / Trunk (push) Successful in 1m19s
Continuous integration / Rustfmt (push) Successful in 1m40s
Continuous integration / build (push) Successful in 2m16s
Continuous integration / Disallow unused dependencies (push) Successful in 5m35s
2026-01-11 11:31:18 -08:00
2d1e38bdff fix(deps): update rust crate quick-xml to 0.39.0
All checks were successful
Continuous integration / Check (push) Successful in 2m33s
Continuous integration / Test Suite (push) Successful in 4m6s
Continuous integration / Rustfmt (push) Successful in 1m38s
Continuous integration / build (push) Successful in 4m17s
Continuous integration / Trunk (push) Successful in 7m52s
Continuous integration / Disallow unused dependencies (push) Successful in 5m40s
2026-01-11 19:01:21 +00:00
33bb355975 chore: Release
All checks were successful
Continuous integration / Check (push) Successful in 1m48s
Continuous integration / Test Suite (push) Successful in 2m4s
Continuous integration / Rustfmt (push) Successful in 58s
Continuous integration / Trunk (push) Successful in 2m11s
Continuous integration / build (push) Successful in 3m33s
Continuous integration / Disallow unused dependencies (push) Successful in 5m42s
2026-01-08 16:13:19 -08:00
2f20a3a8ed server: set user-agent when slurping, fixes /. bug 2026-01-08 16:13:00 -08:00
8b3bfe253f Merge pull request 'chore(deps): update actions/checkout action to v6' (#197) from renovate/actions-checkout-6.x into master
All checks were successful
Continuous integration / Check (push) Successful in 3m43s
Continuous integration / Test Suite (push) Successful in 3m52s
Continuous integration / Rustfmt (push) Successful in 46s
Continuous integration / Trunk (push) Successful in 2m27s
Continuous integration / build (push) Successful in 2m20s
Continuous integration / Disallow unused dependencies (push) Successful in 5m39s
Reviewed-on: #197
2026-01-08 12:23:57 -08:00
22b9646ac4 Merge pull request 'fix(deps): update rust crate zip to v7' (#217) from renovate/zip-7.x into master
Some checks failed
Continuous integration / Check (push) Has been cancelled
Continuous integration / Test Suite (push) Has been cancelled
Continuous integration / Trunk (push) Has been cancelled
Continuous integration / Rustfmt (push) Has been cancelled
Continuous integration / build (push) Has been cancelled
Continuous integration / Disallow unused dependencies (push) Has been cancelled
Reviewed-on: #217
2026-01-08 12:23:42 -08:00
1df8ad8a0c Merge branch 'renovate/all-minor-patch'
All checks were successful
Continuous integration / Check (push) Successful in 1m26s
Continuous integration / Trunk (push) Successful in 59s
Continuous integration / Rustfmt (push) Successful in 49s
Continuous integration / Test Suite (push) Successful in 4m6s
Continuous integration / build (push) Successful in 2m43s
Continuous integration / Disallow unused dependencies (push) Successful in 5m49s
2026-01-08 10:51:25 -08:00
bb7721dbc6 Updates for askama 2026-01-08 10:51:09 -08:00
475c552e3a fix(deps): update all non-major dependencies
Some checks failed
Continuous integration / Check (push) Failing after 1m54s
Continuous integration / Trunk (push) Successful in 1m34s
Continuous integration / Rustfmt (push) Successful in 45s
Continuous integration / Test Suite (push) Failing after 5m28s
Continuous integration / build (push) Failing after 3m29s
Continuous integration / Disallow unused dependencies (push) Failing after 5m41s
2026-01-08 17:03:02 +00:00
c85832c93b chore: Release
All checks were successful
Continuous integration / Check (push) Successful in 1m28s
Continuous integration / Test Suite (push) Successful in 2m30s
Continuous integration / Rustfmt (push) Successful in 1m4s
Continuous integration / Trunk (push) Successful in 2m5s
Continuous integration / build (push) Successful in 2m24s
Continuous integration / Disallow unused dependencies (push) Successful in 5m0s
2026-01-08 08:54:01 -08:00
7e991186fe server: add indexes that should help with bulk read 2026-01-08 08:53:42 -08:00
95d06ec669 Merge pull request 'chore(deps): lock file maintenance' (#224) from renovate/lock-file-maintenance into master
All checks were successful
Continuous integration / Check (push) Successful in 1m18s
Continuous integration / Test Suite (push) Successful in 4m27s
Continuous integration / Rustfmt (push) Successful in 40s
Continuous integration / Trunk (push) Successful in 3m56s
Continuous integration / build (push) Successful in 1m56s
Continuous integration / Disallow unused dependencies (push) Successful in 5m9s
2026-01-04 16:32:18 -08:00
84810d8644 chore(deps): lock file maintenance
All checks were successful
Continuous integration / Check (push) Successful in 2m19s
Continuous integration / Trunk (push) Successful in 1m37s
Continuous integration / Rustfmt (push) Successful in 46s
Continuous integration / build (push) Successful in 4m33s
Continuous integration / Test Suite (push) Successful in 9m6s
Continuous integration / Disallow unused dependencies (push) Successful in 2m21s
2026-01-05 00:02:23 +00:00
8a86f0d0b2 chore: Release
All checks were successful
Continuous integration / Check (push) Successful in 1m8s
Continuous integration / Test Suite (push) Successful in 3m35s
Continuous integration / Rustfmt (push) Successful in 55s
Continuous integration / Trunk (push) Successful in 2m4s
Continuous integration / build (push) Successful in 1m51s
Continuous integration / Disallow unused dependencies (push) Successful in 5m1s
2026-01-04 11:08:22 -08:00
eab4986fd3 server: fix date extraction and unit tests 2026-01-04 11:08:03 -08:00
3c644c570e Merge pull request 'chore(deps): lock file maintenance' (#223) from renovate/lock-file-maintenance into master
All checks were successful
Continuous integration / Check (push) Successful in 1m25s
Continuous integration / Test Suite (push) Successful in 2m17s
Continuous integration / Trunk (push) Successful in 2m5s
Continuous integration / Rustfmt (push) Successful in 42s
Continuous integration / Disallow unused dependencies (push) Successful in 2m26s
Continuous integration / build (push) Successful in 9m7s
2025-12-28 18:46:57 -08:00
7a9df3c15c chore(deps): lock file maintenance
All checks were successful
Continuous integration / Check (push) Successful in 2m28s
Continuous integration / Test Suite (push) Successful in 2m42s
Continuous integration / Rustfmt (push) Successful in 42s
Continuous integration / build (push) Successful in 2m41s
Continuous integration / Disallow unused dependencies (push) Successful in 2m14s
Continuous integration / Trunk (push) Successful in 20m53s
2025-12-29 01:47:41 +00:00
f9d8acf744 Merge pull request 'chore(deps): lock file maintenance' (#222) from renovate/lock-file-maintenance into master
All checks were successful
Continuous integration / Check (push) Successful in 2m14s
Continuous integration / Test Suite (push) Successful in 1m51s
Continuous integration / Rustfmt (push) Successful in 1m2s
Continuous integration / Trunk (push) Successful in 2m7s
Continuous integration / build (push) Successful in 2m7s
Continuous integration / Disallow unused dependencies (push) Successful in 5m4s
2025-12-28 16:48:08 -08:00
75f3770f3e chore(deps): lock file maintenance
All checks were successful
Continuous integration / Check (push) Successful in 4m26s
Continuous integration / Test Suite (push) Successful in 4m25s
Continuous integration / Rustfmt (push) Successful in 40s
Continuous integration / build (push) Successful in 4m10s
Continuous integration / Disallow unused dependencies (push) Successful in 2m18s
Continuous integration / Trunk (push) Successful in 20m54s
2025-12-29 00:02:21 +00:00
85dd61a272 Merge pull request 'chore(deps): update rust crate serde_json to v1.0.146' (#220) from renovate/all-minor-patch into master
All checks were successful
Continuous integration / Check (push) Successful in 1m29s
Continuous integration / Test Suite (push) Successful in 6m57s
Continuous integration / Trunk (push) Successful in 7m51s
Continuous integration / Rustfmt (push) Successful in 1m30s
Continuous integration / build (push) Successful in 2m18s
Continuous integration / Disallow unused dependencies (push) Successful in 5m1s
2025-12-22 05:01:52 -08:00
1c5412de14 chore(deps): update rust crate serde_json to v1.0.146
All checks were successful
Continuous integration / Check (push) Successful in 2m41s
Continuous integration / Test Suite (push) Successful in 2m46s
Continuous integration / Rustfmt (push) Successful in 54s
Continuous integration / build (push) Successful in 2m51s
Continuous integration / Disallow unused dependencies (push) Successful in 2m16s
Continuous integration / Trunk (push) Successful in 21m20s
2025-12-22 12:31:26 +00:00
034027ddd5 Merge pull request 'chore(deps): lock file maintenance' (#219) from renovate/lock-file-maintenance into master
All checks were successful
Continuous integration / Check (push) Successful in 1m39s
Continuous integration / Test Suite (push) Successful in 2m37s
Continuous integration / Trunk (push) Successful in 2m5s
Continuous integration / Rustfmt (push) Successful in 53s
Continuous integration / Disallow unused dependencies (push) Successful in 2m30s
Continuous integration / build (push) Successful in 7m6s
2025-12-21 16:47:07 -08:00
81a07a8172 chore(deps): lock file maintenance
All checks were successful
Continuous integration / Check (push) Successful in 3m28s
Continuous integration / Test Suite (push) Successful in 3m40s
Continuous integration / Rustfmt (push) Successful in 47s
Continuous integration / build (push) Successful in 3m30s
Continuous integration / Disallow unused dependencies (push) Successful in 2m22s
Continuous integration / Trunk (push) Successful in 21m0s
2025-12-22 00:02:30 +00:00
d9e8c2133e Merge pull request 'chore(deps): update rust crate axum to v0.8.8' (#218) from renovate/axum-monorepo into master
All checks were successful
Continuous integration / Check (push) Successful in 1m10s
Continuous integration / Trunk (push) Successful in 1m12s
Continuous integration / Rustfmt (push) Successful in 50s
Continuous integration / Test Suite (push) Successful in 3m2s
Continuous integration / build (push) Successful in 1m54s
Continuous integration / Disallow unused dependencies (push) Successful in 5m0s
2025-12-20 09:02:03 -08:00
99aa7a7071 chore(deps): update rust crate axum to v0.8.8
All checks were successful
Continuous integration / Check (push) Successful in 1m11s
Continuous integration / Test Suite (push) Successful in 3m36s
Continuous integration / Rustfmt (push) Successful in 1m30s
Continuous integration / build (push) Successful in 4m12s
Continuous integration / Trunk (push) Successful in 8m35s
Continuous integration / Disallow unused dependencies (push) Successful in 5m2s
2025-12-20 16:46:32 +00:00
bf7418339e fix(deps): update rust crate zip to v7
All checks were successful
Continuous integration / Check (push) Successful in 1m36s
Continuous integration / Test Suite (push) Successful in 2m46s
Continuous integration / Rustfmt (push) Successful in 40s
Continuous integration / build (push) Successful in 2m46s
Continuous integration / Disallow unused dependencies (push) Successful in 2m19s
Continuous integration / Trunk (push) Successful in 21m7s
2025-12-20 00:01:57 +00:00
cc585cc63f Merge pull request 'chore(deps): lock file maintenance' (#214) from renovate/lock-file-maintenance into master
All checks were successful
Continuous integration / Check (push) Successful in 55s
Continuous integration / Test Suite (push) Successful in 1m17s
Continuous integration / Trunk (push) Successful in 1m3s
Continuous integration / Rustfmt (push) Successful in 1m29s
Continuous integration / build (push) Successful in 1m28s
Continuous integration / Disallow unused dependencies (push) Successful in 5m3s
2025-12-15 09:47:04 -08:00
293f90fde5 chore(deps): lock file maintenance
All checks were successful
Continuous integration / Check (push) Successful in 1m3s
Continuous integration / Test Suite (push) Successful in 1m38s
Continuous integration / Trunk (push) Successful in 1m13s
Continuous integration / Rustfmt (push) Successful in 41s
Continuous integration / build (push) Successful in 1m36s
Continuous integration / Disallow unused dependencies (push) Successful in 2m17s
2025-12-15 17:17:40 +00:00
a06e4b3454 Merge pull request 'chore(deps): update rust crate reqwest to v0.12.26' (#215) from renovate/all-minor-patch into master
All checks were successful
Continuous integration / Check (push) Successful in 56s
Continuous integration / Test Suite (push) Successful in 1m22s
Continuous integration / Rustfmt (push) Successful in 40s
Continuous integration / build (push) Successful in 1m36s
Continuous integration / Disallow unused dependencies (push) Successful in 2m27s
Continuous integration / Trunk (push) Successful in 21m2s
2025-12-15 09:16:55 -08:00
6e5145e21b chore(deps): update rust crate reqwest to v0.12.26
All checks were successful
Continuous integration / Check (push) Successful in 56s
Continuous integration / Test Suite (push) Successful in 1m38s
Continuous integration / Trunk (push) Successful in 1m5s
Continuous integration / Rustfmt (push) Successful in 1m30s
Continuous integration / build (push) Successful in 2m18s
Continuous integration / Disallow unused dependencies (push) Successful in 5m6s
2025-12-15 16:46:42 +00:00
d41f3e9fd1 Merge pull request 'chore(deps): lock file maintenance' (#213) from renovate/lock-file-maintenance into master
All checks were successful
Continuous integration / Check (push) Successful in 58s
Continuous integration / Test Suite (push) Successful in 1m37s
Continuous integration / Rustfmt (push) Successful in 54s
Continuous integration / Trunk (push) Successful in 2m3s
Continuous integration / build (push) Successful in 1m42s
Continuous integration / Disallow unused dependencies (push) Successful in 5m4s
2025-12-14 16:47:42 -08:00
5519018043 chore(deps): lock file maintenance
All checks were successful
Continuous integration / Check (push) Successful in 1m15s
Continuous integration / Test Suite (push) Successful in 2m23s
Continuous integration / Rustfmt (push) Successful in 1m16s
Continuous integration / build (push) Successful in 2m42s
Continuous integration / Disallow unused dependencies (push) Successful in 2m24s
Continuous integration / Trunk (push) Successful in 21m15s
2025-12-15 00:02:24 +00:00
e3121219b6 chore: Release
All checks were successful
Continuous integration / Check (push) Successful in 1m3s
Continuous integration / Test Suite (push) Successful in 1m19s
Continuous integration / Trunk (push) Successful in 1m2s
Continuous integration / Rustfmt (push) Successful in 49s
Continuous integration / Disallow unused dependencies (push) Successful in 2m26s
Continuous integration / build (push) Successful in 3m12s
2025-12-14 09:14:13 -08:00
7272bbb6b0 web: text overflow at 2 lines for subject and authors 2025-12-14 09:13:48 -08:00
dc741f421b Merge pull request 'fix(deps): update all non-major dependencies' (#212) from renovate/all-minor-patch into master
All checks were successful
Continuous integration / Check (push) Successful in 1m26s
Continuous integration / Trunk (push) Successful in 1m11s
Continuous integration / Test Suite (push) Successful in 3m17s
Continuous integration / Rustfmt (push) Successful in 42s
Continuous integration / Disallow unused dependencies (push) Successful in 2m25s
Continuous integration / build (push) Successful in 7m21s
2025-12-08 13:17:00 -08:00
69d3b8a210 fix(deps): update all non-major dependencies
All checks were successful
Continuous integration / Check (push) Successful in 2m9s
Continuous integration / Test Suite (push) Successful in 2m50s
Continuous integration / Rustfmt (push) Successful in 1m27s
Continuous integration / build (push) Successful in 3m50s
Continuous integration / Disallow unused dependencies (push) Successful in 2m33s
Continuous integration / Trunk (push) Successful in 20m56s
2025-12-08 20:32:37 +00:00
f5c4067291 Merge pull request 'chore(deps): lock file maintenance' (#211) from renovate/lock-file-maintenance into master
All checks were successful
Continuous integration / Check (push) Successful in 1m28s
Continuous integration / Test Suite (push) Successful in 2m17s
Continuous integration / Trunk (push) Successful in 7m39s
Continuous integration / Rustfmt (push) Successful in 1m29s
Continuous integration / build (push) Successful in 3m25s
Continuous integration / Disallow unused dependencies (push) Successful in 5m7s
2025-12-07 16:32:06 -08:00
930a45cbad chore(deps): lock file maintenance
All checks were successful
Continuous integration / Check (push) Successful in 2m21s
Continuous integration / Trunk (push) Successful in 2m41s
Continuous integration / Test Suite (push) Successful in 4m28s
Continuous integration / Rustfmt (push) Successful in 1m30s
Continuous integration / build (push) Successful in 4m42s
Continuous integration / Disallow unused dependencies (push) Successful in 5m7s
2025-12-08 00:02:19 +00:00
ef612c0d4f Merge pull request 'fix(deps): update rust crate scraper to 0.25.0' (#210) from renovate/all-minor-patch into master
All checks were successful
Continuous integration / Check (push) Successful in 1m59s
Continuous integration / Trunk (push) Successful in 2m8s
Continuous integration / Test Suite (push) Successful in 4m51s
Continuous integration / Rustfmt (push) Successful in 1m30s
Continuous integration / build (push) Successful in 2m37s
Continuous integration / Disallow unused dependencies (push) Successful in 5m5s
2025-12-07 14:46:27 -08:00
723e9c5ff5 fix(deps): update rust crate scraper to 0.25.0
All checks were successful
Continuous integration / Check (push) Successful in 2m57s
Continuous integration / Test Suite (push) Successful in 2m13s
Continuous integration / Rustfmt (push) Successful in 43s
Continuous integration / build (push) Successful in 2m15s
Continuous integration / Disallow unused dependencies (push) Successful in 2m44s
Continuous integration / Trunk (push) Successful in 21m9s
2025-12-07 22:16:57 +00:00
0fdcfabfbe Merge pull request 'chore(deps): update rust crate html2text to v0.16.5' (#209) from renovate/all-minor-patch into master
All checks were successful
Continuous integration / Check (push) Successful in 1m57s
Continuous integration / Test Suite (push) Successful in 2m50s
Continuous integration / Trunk (push) Successful in 1m17s
Continuous integration / Rustfmt (push) Successful in 1m28s
Continuous integration / build (push) Successful in 2m16s
Continuous integration / Disallow unused dependencies (push) Successful in 5m1s
2025-12-06 00:31:45 -08:00
b6c3f014cb chore(deps): update rust crate html2text to v0.16.5
All checks were successful
Continuous integration / Check (push) Successful in 1m30s
Continuous integration / Test Suite (push) Successful in 2m50s
Continuous integration / Rustfmt (push) Successful in 1m31s
Continuous integration / Trunk (push) Successful in 8m56s
Continuous integration / build (push) Successful in 6m26s
Continuous integration / Disallow unused dependencies (push) Successful in 2m52s
2025-12-06 08:17:07 +00:00
1937bb4c99 Merge pull request 'chore(deps): update rust crate flate2 to v1.1.7' (#208) from renovate/all-minor-patch into master
Some checks failed
Continuous integration / Check (push) Failing after 12s
Continuous integration / Trunk (push) Failing after 11s
Continuous integration / Rustfmt (push) Failing after 12s
Continuous integration / build (push) Failing after 11s
Continuous integration / Disallow unused dependencies (push) Failing after 11s
Continuous integration / Test Suite (push) Successful in 2m19s
2025-12-05 02:31:21 -08:00
cdd5d9befc chore(deps): update rust crate flate2 to v1.1.7
All checks were successful
Continuous integration / Check (push) Successful in 1m43s
Continuous integration / Trunk (push) Successful in 1m9s
Continuous integration / Rustfmt (push) Successful in 44s
Continuous integration / build (push) Successful in 2m32s
Continuous integration / Disallow unused dependencies (push) Successful in 2m18s
Continuous integration / Test Suite (push) Successful in 9m35s
2025-12-05 10:16:24 +00:00
232a14fd96 Merge pull request 'chore(deps): update rust crate log to v0.4.29' (#206) from renovate/all-minor-patch into master
All checks were successful
Continuous integration / Check (push) Successful in 1m19s
Continuous integration / Test Suite (push) Successful in 2m6s
Continuous integration / Trunk (push) Successful in 1m4s
Continuous integration / Rustfmt (push) Successful in 43s
Continuous integration / build (push) Successful in 2m23s
Continuous integration / Disallow unused dependencies (push) Successful in 2m28s
2025-12-02 14:46:36 -08:00
3038c98a7a chore(deps): update rust crate log to v0.4.29
All checks were successful
Continuous integration / Check (push) Successful in 1m31s
Continuous integration / Test Suite (push) Successful in 3m40s
Continuous integration / Trunk (push) Successful in 1m32s
Continuous integration / Rustfmt (push) Successful in 46s
Continuous integration / build (push) Successful in 4m34s
Continuous integration / Disallow unused dependencies (push) Successful in 2m33s
2025-12-02 22:16:32 +00:00
4dd240c358 Merge pull request 'chore(deps): update rust crate uuid to v1.19.0' (#205) from renovate/all-minor-patch into master
All checks were successful
Continuous integration / Check (push) Successful in 1m12s
Continuous integration / Test Suite (push) Successful in 2m58s
Continuous integration / Trunk (push) Successful in 1m13s
Continuous integration / Rustfmt (push) Successful in 41s
Continuous integration / build (push) Successful in 3m4s
Continuous integration / Disallow unused dependencies (push) Successful in 2m22s
2025-12-01 19:31:17 -08:00
c66e876ab7 chore(deps): update rust crate uuid to v1.19.0
All checks were successful
Continuous integration / Check (push) Successful in 1m28s
Continuous integration / Test Suite (push) Successful in 2m13s
Continuous integration / Trunk (push) Successful in 7m37s
Continuous integration / build (push) Successful in 2m3s
Continuous integration / Disallow unused dependencies (push) Successful in 2m35s
Continuous integration / Rustfmt (push) Successful in 43s
2025-12-02 02:46:26 +00:00
a7762595fa Merge pull request 'chore(deps): lock file maintenance' (#204) from renovate/lock-file-maintenance into master
All checks were successful
Continuous integration / Check (push) Successful in 1m20s
Continuous integration / Test Suite (push) Successful in 2m30s
Continuous integration / Trunk (push) Successful in 7m27s
Continuous integration / Rustfmt (push) Successful in 42s
Continuous integration / build (push) Successful in 2m28s
Continuous integration / Disallow unused dependencies (push) Successful in 2m24s
2025-11-30 16:46:57 -08:00
1ac471dfe7 chore(deps): lock file maintenance
All checks were successful
Continuous integration / Check (push) Successful in 1m46s
Continuous integration / Test Suite (push) Successful in 3m26s
Continuous integration / Trunk (push) Successful in 7m52s
Continuous integration / Rustfmt (push) Successful in 42s
Continuous integration / build (push) Successful in 3m26s
Continuous integration / Disallow unused dependencies (push) Successful in 2m20s
2025-12-01 00:01:57 +00:00
72a549ea0f Merge pull request 'chore(deps): update rust crate tracing to v0.1.43' (#203) from renovate/tokio-tracing-monorepo into master
All checks were successful
Continuous integration / Check (push) Successful in 1m46s
Continuous integration / Test Suite (push) Successful in 2m49s
Continuous integration / Trunk (push) Successful in 1m8s
Continuous integration / Rustfmt (push) Successful in 56s
Continuous integration / build (push) Successful in 2m18s
Continuous integration / Disallow unused dependencies (push) Successful in 2m58s
2025-11-28 01:31:57 -08:00
878afd695f chore(deps): update rust crate tracing to v0.1.43
All checks were successful
Continuous integration / Check (push) Successful in 1m29s
Continuous integration / Test Suite (push) Successful in 3m3s
Continuous integration / Trunk (push) Successful in 7m58s
Continuous integration / Rustfmt (push) Successful in 53s
Continuous integration / build (push) Successful in 2m53s
Continuous integration / Disallow unused dependencies (push) Successful in 2m20s
2025-11-28 09:01:41 +00:00
50b23731df Merge pull request 'chore(deps): update rust crate xtracing to v0.3.3' (#202) from renovate/all-minor-patch into master
All checks were successful
Continuous integration / Check (push) Successful in 1m14s
Continuous integration / Test Suite (push) Successful in 1m57s
Continuous integration / Trunk (push) Successful in 2m7s
Continuous integration / Rustfmt (push) Successful in 46s
Continuous integration / build (push) Successful in 2m17s
Continuous integration / Disallow unused dependencies (push) Successful in 2m50s
2025-11-27 16:18:03 -08:00
95df6b54ea chore(deps): update rust crate xtracing to v0.3.3
All checks were successful
Continuous integration / Check (push) Successful in 1m13s
Continuous integration / Test Suite (push) Successful in 2m29s
Continuous integration / Trunk (push) Successful in 1m8s
Continuous integration / Rustfmt (push) Successful in 44s
Continuous integration / build (push) Successful in 3m0s
Continuous integration / Disallow unused dependencies (push) Successful in 2m50s
2025-11-27 23:32:15 +00:00
ee626eb631 chore: Release
All checks were successful
Continuous integration / Check (push) Successful in 1m21s
Continuous integration / Test Suite (push) Successful in 2m5s
Continuous integration / Trunk (push) Successful in 58s
Continuous integration / Rustfmt (push) Successful in 43s
Continuous integration / build (push) Successful in 1m57s
Continuous integration / Disallow unused dependencies (push) Successful in 2m26s
2025-11-27 10:11:04 -08:00
26f805738d web: fix progress on news + catchup mode
Some checks failed
Continuous integration / Check (push) Has been cancelled
Continuous integration / Test Suite (push) Has been cancelled
Continuous integration / Trunk (push) Has been cancelled
Continuous integration / Rustfmt (push) Has been cancelled
Continuous integration / build (push) Has been cancelled
Continuous integration / Disallow unused dependencies (push) Has been cancelled
2025-11-27 10:10:42 -08:00
51ff0b8e14 chore(deps): update actions/checkout action to v6
All checks were successful
Continuous integration / Check (push) Successful in 58s
Continuous integration / Test Suite (push) Successful in 1m51s
Continuous integration / Rustfmt (push) Successful in 41s
Continuous integration / build (push) Successful in 1m53s
Continuous integration / Disallow unused dependencies (push) Successful in 2m19s
Continuous integration / Trunk (push) Successful in 1m41s
2025-11-20 16:46:50 +00:00
12 changed files with 2682 additions and 719 deletions

View File

@@ -7,7 +7,7 @@ jobs:
name: Check
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: actions/checkout@v6
- uses: actions-rust-lang/setup-rust-toolchain@v1
- run: cargo check
@@ -15,7 +15,7 @@ jobs:
name: Test Suite
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: actions/checkout@v6
- uses: actions-rust-lang/setup-rust-toolchain@v1
- run: cargo test
@@ -23,7 +23,7 @@ jobs:
name: Trunk
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: actions/checkout@v6
- uses: actions-rust-lang/setup-rust-toolchain@v1
with:
toolchain: nightly
@@ -35,7 +35,7 @@ jobs:
name: Rustfmt
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: actions/checkout@v6
- uses: actions-rust-lang/setup-rust-toolchain@v1
with:
components: rustfmt
@@ -46,7 +46,7 @@ jobs:
name: build
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: actions/checkout@v6
- uses: actions-rust-lang/setup-rust-toolchain@v1
- run: cargo build
@@ -54,7 +54,7 @@ jobs:
name: Disallow unused dependencies
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: actions/checkout@v6
- uses: actions-rust-lang/setup-rust-toolchain@v1
with:
toolchain: nightly

1605
Cargo.lock generated

File diff suppressed because it is too large Load Diff

View File

@@ -8,7 +8,7 @@ authors = ["Bill Thiede <git@xinu.tv>"]
edition = "2021"
license = "UNLICENSED"
publish = ["xinu"]
version = "0.17.53"
version = "0.17.61"
repository = "https://git.z.xinu.tv/wathiede/letterbox"
[profile.dev]

View File

@@ -16,7 +16,7 @@ chrono-tz = "0.10"
html2text = "0.16"
ammonia = "4.1.0"
anyhow = "1.0.98"
askama = { version = "0.14.0", features = ["derive"] }
askama = { version = "0.15.0", features = ["derive"] }
async-graphql = { version = "7", features = ["log", "chrono"] }
async-graphql-axum = "7.0.16"
async-trait = "0.1.88"
@@ -26,7 +26,7 @@ build-info = "0.0.42"
cacher = { version = "0.2.0", registry = "xinu" }
chrono = "0.4.40"
clap = { version = "4.5.37", features = ["derive"] }
css-inline = "0.18.0"
css-inline = "0.19.0"
flate2 = "1.1.2"
futures = "0.3.31"
headers = "0.4.0"
@@ -39,10 +39,10 @@ lol_html = "2.3.0"
mailparse = "0.16.1"
maplit = "1.0.2"
memmap = "0.7.0"
quick-xml = { version = "0.38.1", features = ["serialize"] }
quick-xml = { version = "0.39.0", features = ["serialize"] }
regex = "1.11.1"
reqwest = { version = "0.12.15", features = ["blocking"] }
scraper = "0.24.0"
reqwest = { version = "0.13.0", features = ["blocking"] }
scraper = "0.25.0"
serde = { version = "1.0.219", features = ["derive"] }
serde_json = "1.0.140"
sqlx = { version = "0.8.5", features = ["postgres", "runtime-tokio", "chrono"] }
@@ -56,7 +56,7 @@ urlencoding = "2.1.3"
#xtracing = { git = "http://git-private.h.xinu.tv/wathiede/xtracing.git" }
#xtracing = { path = "../../xtracing" }
xtracing = { version = "0.3.2", registry = "xinu" }
zip = "6.0.0"
zip = "7.0.0"
[build-dependencies]

View File

@@ -0,0 +1,2 @@
DROP INDEX IF EXISTS movie_sets_year_id_idx;
DROP INDEX IF EXISTS movie_sets_year_idx;

View File

@@ -0,0 +1,6 @@
-- Add index on movie_sets.year to speed up year-based queries
CREATE INDEX movie_sets_year_idx ON movie_sets(year);
-- Composite index for queries that filter by year and return id
-- This can make the subquery in UPDATE statements even faster
CREATE INDEX movie_sets_year_id_idx ON movie_sets(year, id);

View File

@@ -34,6 +34,7 @@ const TEXT_PLAIN: &'static str = "text/plain";
// Inline Askama filters module for template use
mod filters {
// Usage: {{ items|batch(7) }}
#[askama::filter_fn]
pub fn batch<T: Clone>(
items: &[T],
_: &dyn ::askama::Values,
@@ -148,7 +149,7 @@ pub fn extract_calendar_metadata_from_mail(
// Fallback extraction: if iCal did not provide metadata, extract from subject/body before generating fallback HTML
if body_html.is_none() {
// Try to extract summary from subject (e.g., "New event: <summary> @ ...")
// Try to extract summary from subject (e.g., "New event: <summary> @ ..." or "Updated invitation: <summary> @ ...")
if summary.is_none() {
if let Some(subject) = m.headers.get_first_value("Subject") {
if let Some(caps) = regex::Regex::new(r"New event: ([^@]+) @")
@@ -161,22 +162,52 @@ pub fn extract_calendar_metadata_from_mail(
.and_then(|re| re.captures(&subject))
{
summary = Some(caps[1].trim().to_string());
} else if let Some(caps) = regex::Regex::new(r"Updated invitation: ([^@]+) @")
.ok()
.and_then(|re| re.captures(&subject))
{
summary = Some(caps[1].trim().to_string());
}
}
}
// Try to extract start/end dates from subject
if start_date.is_none() || end_date.is_none() {
if let Some(subject) = m.headers.get_first_value("Subject") {
// Pattern: New event: Dentist appt @ Tue Sep 23, 2025 3pm - 4pm (PDT) (tconvertino@gmail.com)
if let Some(caps) = regex::Regex::new(r"New event: [^@]+@ ([A-Za-z]{3}) ([A-Za-z]{3}) (\d{1,2}), (\d{4}) (\d{1,2})(?::(\d{2}))? ?([ap]m) ?- ?(\d{1,2})(?::(\d{2}))? ?([ap]m)").ok().and_then(|re| re.captures(&subject)) {
// Pattern: @ Tue Sep 23, 2025 3pm - 4pm (works for New event, Invitation, Updated invitation, etc.)
if let Some(caps) = regex::Regex::new(r"@ ([A-Za-z]{3}) ([A-Za-z]{3}) (\d{1,2}), (\d{4}) (\d{1,2})(?::(\d{2}))? ?([ap]m) ?- ?(\d{1,2})(?::(\d{2}))? ?([ap]m)").ok().and_then(|re| re.captures(&subject)) {
let month = &caps[2];
let day = &caps[3];
let year = &caps[4];
let start_hour: u32 = caps[5].parse().unwrap_or(0);
let start_min: u32 = caps.get(6).map(|m| m.as_str().parse().unwrap_or(0)).unwrap_or(0);
let start_ampm = &caps[7];
let end_hour: u32 = caps[8].parse().unwrap_or(0);
let end_min: u32 = caps.get(9).map(|m| m.as_str().parse().unwrap_or(0)).unwrap_or(0);
let end_ampm = &caps[10];
// Convert 12-hour to 24-hour format
let start_hour_24 = if start_ampm == "pm" && start_hour != 12 {
start_hour + 12
} else if start_ampm == "am" && start_hour == 12 {
0
} else {
start_hour
};
let end_hour_24 = if end_ampm == "pm" && end_hour != 12 {
end_hour + 12
} else if end_ampm == "am" && end_hour == 12 {
0
} else {
end_hour
};
let date_str = format!("{} {} {}", month, day, year);
if let Ok(date) = chrono::NaiveDate::parse_from_str(&date_str, "%b %d %Y") {
let ymd = date.format("%Y%m%d").to_string();
start_date = Some(ymd.clone());
end_date = Some(ymd);
// Store date with time in YYYYMMDDTHHMMSS format for start/end
let start_dt = format!("{}T{:02}{:02}00", date.format("%Y%m%d"), start_hour_24, start_min);
let end_dt = format!("{}T{:02}{:02}00", date.format("%Y%m%d"), end_hour_24, end_min);
start_date = Some(start_dt);
end_date = Some(end_dt);
}
} else {
// Pattern: from Thu Sep 11 to Fri Jan 30, 2026
@@ -195,8 +226,16 @@ pub fn extract_calendar_metadata_from_mail(
}
}
if let (Some(sm), Some(em)) = (month_num(start_month), month_num(end_month)) {
let current_year = chrono::Local::now().year().to_string();
let start = format!("{}{}{}", current_year, sm, format!("{:0>2}", start_day));
// If start month is later in calendar year than end month, start is in previous year
let sm_num: u32 = sm.parse().unwrap_or(1);
let em_num: u32 = em.parse().unwrap_or(1);
let end_year = year.parse::<i32>().unwrap_or_else(|_| chrono::Local::now().year());
let start_year: i32 = if sm_num > em_num {
end_year - 1
} else {
end_year
};
let start = format!("{}{}{}", start_year, sm, format!("{:0>2}", start_day));
let mut end_date_val = chrono::NaiveDate::parse_from_str(&format!("{}-{}-{}", year, em, format!("{:0>2}", end_day)), "%Y-%m-%d").ok();
if let Some(d) = end_date_val.as_mut() {
*d = d.succ_opt().unwrap_or(*d);
@@ -232,6 +271,31 @@ pub fn extract_calendar_metadata_from_mail(
if end_date.is_none() { end_date = Some(end); }
}
}
// Pattern: single all-day event: @ Sun Jan 18, 2026 (no time range)
if start_date.is_none() {
if let Some(caps) = regex::Regex::new(r"@ [A-Za-z]{3} ([A-Za-z]{3}) (\d{1,2}), (\d{4})(?:\s*\(|$)").ok().and_then(|re| re.captures(&subject)) {
let month = &caps[1];
let day = &caps[2];
let year = &caps[3];
fn month_num(mon: &str) -> Option<&'static str> {
match mon {
"Jan" => Some("01"), "Feb" => Some("02"), "Mar" => Some("03"), "Apr" => Some("04"),
"May" => Some("05"), "Jun" => Some("06"), "Jul" => Some("07"), "Aug" => Some("08"),
"Sep" => Some("09"), "Oct" => Some("10"), "Nov" => Some("11"), "Dec" => Some("12"),
_ => None
}
}
if let Some(mm) = month_num(month) {
let start = format!("{}{}{:0>2}", year, mm, day);
// For all-day events, end date is the next day (exclusive)
if let Ok(d) = chrono::NaiveDate::parse_from_str(&format!("{}-{}-{:0>2}", year, mm, day), "%Y-%m-%d") {
let end = d.succ_opt().unwrap_or(d).format("%Y%m%d").to_string();
start_date = Some(start);
end_date = Some(end);
}
}
}
}
}
}
}
@@ -319,6 +383,8 @@ pub fn extract_calendar_metadata_from_mail(
let needs_ical_flex =
summary.is_some() || start_date.is_some() || end_date.is_some() || has_recurrence;
if needs_ical_flex {
use chrono::{Datelike, NaiveDate};
let summary_val = summary.clone().unwrap_or_default();
let organizer_val = organizer.clone().unwrap_or_default();
let start_val = start_date.clone().unwrap_or_default();
@@ -328,15 +394,133 @@ pub fn extract_calendar_metadata_from_mail(
} else {
String::new()
};
// Compute event_days and all_days for calendar grid rendering
let mut event_days: Vec<NaiveDate> = Vec::new();
let (local_fmt_start, local_fmt_end) = if let (Some(ref start_str), Some(ref end_str)) =
(&start_date, &end_date)
{
// Parse dates - try YYYYMMDDTHHMMSS format first (with time), then YYYYMMDD (date only)
let (start_d, start_time) = if start_str.contains('T') {
let parts: Vec<&str> = start_str.split('T').collect();
let date = NaiveDate::parse_from_str(parts[0], "%Y%m%d").ok();
let time = if parts.len() > 1 {
chrono::NaiveTime::parse_from_str(parts[1], "%H%M%S").ok()
} else {
None
};
(date, time)
} else {
(NaiveDate::parse_from_str(start_str, "%Y%m%d").ok(), None)
};
let (end_d, end_time) = if end_str.contains('T') {
let parts: Vec<&str> = end_str.split('T').collect();
let date = NaiveDate::parse_from_str(parts[0], "%Y%m%d").ok();
let time = if parts.len() > 1 {
chrono::NaiveTime::parse_from_str(parts[1], "%H%M%S").ok()
} else {
None
};
(date, time)
} else {
(NaiveDate::parse_from_str(end_str, "%Y%m%d").ok(), None)
};
if let (Some(start), Some(end)) = (start_d, end_d) {
// For all-day events (no time), end date is exclusive, so we need to subtract one day
let is_allday = start_time.is_none() && end_time.is_none();
let display_end = if is_allday && end > start {
end.pred_opt().unwrap_or(end)
} else {
end
};
// Add all days from start to display_end (inclusive) to event_days
let mut day_iter = start;
while day_iter <= display_end {
event_days.push(day_iter);
day_iter = day_iter.succ_opt().unwrap_or(day_iter);
if day_iter == display_end && day_iter == start {
// Single day event
break;
}
}
// Format dates for display - include time if available
let fmt_start = if let Some(t) = start_time {
format!("{} {}", start.format("%a %b %e, %Y"), t.format("%-I:%M %p"))
} else {
start.format("%a %b %e, %Y").to_string()
};
let fmt_end = if let Some(t) = end_time {
format!("{} {}", display_end.format("%a %b %e, %Y"), t.format("%-I:%M %p"))
} else {
display_end.format("%a %b %e, %Y").to_string()
};
(fmt_start, fmt_end)
} else {
(start_val.clone(), end_val.clone())
}
} else {
(start_val.clone(), end_val.clone())
};
// Compute calendar grid (all_days) from event_days
let (all_days, caption) = if !event_days.is_empty() {
let first_event = event_days.first().unwrap();
let last_event = event_days.last().unwrap();
let first_of_month =
NaiveDate::from_ymd_opt(first_event.year(), first_event.month(), 1).unwrap();
let last_of_month = {
let next_month = if last_event.month() == 12 {
NaiveDate::from_ymd_opt(last_event.year() + 1, 1, 1).unwrap()
} else {
NaiveDate::from_ymd_opt(last_event.year(), last_event.month() + 1, 1).unwrap()
};
next_month.pred_opt().unwrap()
};
// Start from Sunday of the week containing first_of_month
let mut cal_start = first_of_month;
while cal_start.weekday() != chrono::Weekday::Sun {
cal_start = cal_start.pred_opt().unwrap();
}
// End on Saturday of the week containing last_of_month
let mut cal_end = last_of_month;
while cal_end.weekday() != chrono::Weekday::Sat {
cal_end = cal_end.succ_opt().unwrap();
}
let mut all_days = vec![];
let mut d = cal_start;
while d <= cal_end {
all_days.push(d);
d = d.succ_opt().unwrap();
}
let start_month = first_event.format("%B %Y");
let end_month = last_event.format("%B %Y");
let caption = if start_month.to_string() == end_month.to_string() {
start_month.to_string()
} else {
format!("{} {}", start_month, end_month)
};
(all_days, caption)
} else {
(vec![], String::new())
};
let template = IcalSummaryTemplate {
summary: &summary_val,
local_fmt_start: &start_val,
local_fmt_end: &end_val,
local_fmt_start: &local_fmt_start,
local_fmt_end: &local_fmt_end,
organizer: &organizer_val,
organizer_cn: "",
all_days: vec![],
event_days: vec![],
caption: String::new(),
all_days,
event_days,
caption,
description_paragraphs: &[],
today: Some(chrono::Local::now().date_naive()),
recurrence_display,
@@ -2268,17 +2452,18 @@ mod tests {
assert_eq!(meta.summary, Some("Dentist appt".to_string()));
// Organizer: from From header, extract email address
assert_eq!(meta.organizer, Some("tconvertino@gmail.com".to_string()));
// Dates: should extract Sep 23, 2025, 3pm-4pm
assert_eq!(meta.start_date, Some("20250923".to_string()));
assert_eq!(meta.end_date, Some("20250923".to_string()));
// Dates: should extract Sep 23, 2025, 3pm-4pm (15:00-16:00)
assert_eq!(meta.start_date, Some("20250923T150000".to_string()));
assert_eq!(meta.end_date, Some("20250923T160000".to_string()));
// Should not be recurring
if let Some(ref html) = meta.body_html {
assert!(
html.contains("Dentist appt"),
"HTML should contain the summary"
);
// Date is now formatted as human-readable "Tue Sep 23, 2025"
assert!(
html.contains("20250923"),
html.contains("Sep 23, 2025") || html.contains("20250923"),
"HTML should contain the event date"
);
assert!(
@@ -2307,8 +2492,8 @@ mod tests {
Some("calendar-notification@google.com".to_string())
);
// Dates: from subject, Thu Sep 11 to Fri Jan 30, 2026
let current_year = chrono::Local::now().year();
assert_eq!(meta.start_date, Some(format!("{}0911", current_year)));
// Start date is Sep 11, 2025 (one year before end since Sep > Jan)
assert_eq!(meta.start_date, Some("20250911".to_string()));
assert_eq!(meta.end_date, Some("20260131".to_string()));
}
#[test]
@@ -2332,12 +2517,13 @@ mod tests {
html.contains("<b>Organizer:</b> calendar-notification@google.com"),
"HTML should contain the labeled organizer"
);
// Dates are now formatted as human-readable
assert!(
html.contains("<b>Start:</b> 20250911"),
html.contains("<b>Start:</b> Thu Sep 11, 2025") || html.contains("<b>Start:</b> 20250911"),
"HTML should contain the labeled start time"
);
assert!(
html.contains("<b>End:</b> 20260131"),
html.contains("<b>End:</b> Fri Jan 30, 2026") || html.contains("<b>End:</b> 20260131"),
"HTML should contain the labeled end time"
);
if !html.contains("ical-flex") {
@@ -2408,8 +2594,8 @@ mod tests {
Some("calendar-notification@google.com".to_string())
);
// Dates: from subject, Thu Sep 11 to Fri Jan 30, 2026
let current_year = chrono::Local::now().year();
assert_eq!(meta.start_date, Some(format!("{}0911", current_year)));
// Start date is Sep 11, 2025 (one year before end since Sep > Jan)
assert_eq!(meta.start_date, Some("20250911".to_string()));
assert_eq!(meta.end_date, Some("20260131".to_string()));
// Debug: print the rendered HTML for inspection
if let Some(ref html) = meta.body_html {
@@ -2442,8 +2628,8 @@ mod tests {
Some("calendar-notification@google.com".to_string())
);
// Assert that the start and end dates are present
let current_year = chrono::Local::now().year();
assert_eq!(meta.start_date, Some(format!("{}0911", current_year)));
// Start date is Sep 11, 2025 (one year before end since Sep > Jan)
assert_eq!(meta.start_date, Some("20250911".to_string()));
assert_eq!(meta.end_date, Some("20260131".to_string()));
// Assert that the HTML body contains recurrence info
if let Some(ref html) = meta.body_html {
@@ -2459,6 +2645,56 @@ mod tests {
}
}
#[test]
fn google_calendar_email_4_single_allday_event() {
use mailparse::parse_mail;
let raw_email = include_str!("../../server/testdata/google-calendar-example-4.eml");
let parsed = parse_mail(raw_email.as_bytes()).expect("parse_mail");
let mut part_addr = vec![];
let body = extract_body(&parsed, &mut part_addr).expect("extract_body");
let meta = extract_calendar_metadata_from_mail(&parsed, &body);
// Assert detection as Google Calendar
assert!(meta.is_google_calendar_event);
// Assert metadata extraction for single all-day event
assert_eq!(meta.summary, Some("Emery Sleeps Over".to_string()));
assert_eq!(meta.organizer, Some("tconvertino@gmail.com".to_string()));
// Dates: Sunday Jan 18, 2026 (all-day event)
assert_eq!(meta.start_date, Some("20260118".to_string()));
assert_eq!(meta.end_date, Some("20260119".to_string())); // All-day events end next day
// Assert ical summary is rendered and shows Jan 18 highlighted
let html = meta.body_html.expect("body_html");
println!("Rendered HTML: {}", html);
assert!(html.contains("ical-flex"), "Calendar widget should be rendered");
assert!(html.contains(r#"data-event-day="2026-01-18""#), "Jan 18 should be highlighted");
}
#[test]
fn google_calendar_email_5_updated_invitation() {
use mailparse::parse_mail;
let raw_email = include_str!("../../server/testdata/google-calendar-example-5.eml");
let parsed = parse_mail(raw_email.as_bytes()).expect("parse_mail");
let mut part_addr = vec![];
let body = extract_body(&parsed, &mut part_addr).expect("extract_body");
let meta = extract_calendar_metadata_from_mail(&parsed, &body);
// Assert detection as Google Calendar
assert!(meta.is_google_calendar_event);
// Assert metadata extraction for updated invitation
assert_eq!(meta.summary, Some("painting class".to_string()));
assert_eq!(meta.organizer, Some("tconvertino@gmail.com".to_string()));
// Dates: Thursday Feb 12, 2026 7pm - 9pm (same day event with time)
// Start: 7pm = 19:00, End: 9pm = 21:00
assert_eq!(meta.start_date, Some("20260212T190000".to_string()));
assert_eq!(meta.end_date, Some("20260212T210000".to_string()));
// Assert ical summary is rendered and shows Feb 12 highlighted
let html = meta.body_html.expect("body_html");
println!("Rendered HTML: {}", html);
assert!(html.contains("ical-flex"), "Calendar widget should be rendered");
assert!(html.contains(r#"data-event-day="2026-02-12""#), "Feb 12 should be highlighted");
// Verify time is displayed in the HTML
assert!(html.contains("7:00 PM") || html.contains("7pm") || html.contains("19:00"),
"HTML should contain the start time");
}
#[test]
fn recurring_event_rrule_metadata_and_highlight() {
use super::render_ical_summary;

View File

@@ -44,6 +44,8 @@ use crate::{
const NEWSREADER_TAG_PREFIX: &'static str = "News/";
const NEWSREADER_THREAD_PREFIX: &'static str = "news:";
const USER_AGENT: &'static str = "letterbox news reader (letterbox-ua@xinu.tv)";
// TODO: figure out how to use Cow
#[async_trait]
trait Transformer: Send + Sync {
@@ -318,7 +320,8 @@ impl<'c> Transformer for SlurpContents<'c> {
let body = if let Some(body) = cacher.get(link.as_str()) {
String::from_utf8_lossy(&body).to_string()
} else {
let resp = reqwest::get(link.as_str()).await?;
let client = reqwest::Client::builder().user_agent(USER_AGENT).build()?;
let resp = client.get(link.as_str()).send().await?;
let status = resp.status();
if status.is_server_error() {
error!("status error for {link}: {status}");

View File

@@ -0,0 +1,728 @@
Return-Path: <couchmoney+caf_=gmail=xinu.tv@gmail.com>
Delivered-To: bill@xinu.tv
Received: from phx.xinu.tv [74.207.253.222]
by nixos-01.h.xinu.tv with IMAP (fetchmail-6.5.6)
for <wathiede@localhost> (single-drop); Sat, 17 Jan 2026 09:25:58 -0800 (PST)
Received: from phx.xinu.tv
by phx.xinu.tv with LMTP
id YIkZKyXGa2k93xMAJR8clQ
(envelope-from <couchmoney+caf_=gmail=xinu.tv@gmail.com>)
for <bill@xinu.tv>; Sat, 17 Jan 2026 09:25:57 -0800
X-Original-To: gmail@xinu.tv
Received: from mail-lf1-f48.google.com (mail-lf1-f48.google.com [209.85.167.48])
by phx.xinu.tv (Postfix) with ESMTPS id B744880023
for <gmail@xinu.tv>; Sat, 17 Jan 2026 09:25:56 -0800 (PST)
Received: by mail-lf1-f48.google.com with SMTP id 2adb3069b0e04-59b78886454so3800941e87.2
for <gmail@xinu.tv>; Sat, 17 Jan 2026 09:25:56 -0800 (PST)
ARC-Seal: i=2; a=rsa-sha256; t=1768670755; cv=pass;
d=google.com; s=arc-20240605;
b=UCMG36NoEclyVlwzV5KDOA6Fq75afR1kZ6QZQ8A0CR9RJMMEnPEpiuhheiGH7csZWs
HEZJmrLtTX/e5qiZ0k5njtm8694d+44YtpWRS54bwcAvwWBeCnHstTFkuOB4J2GWvT6G
R9MwX2lwlaGj118bn6aIQTWLB6KyWzUmGdq9AO52fvWTkzlPFDN54/AUYdhx4r+dG5k3
tqmDhE87DYIPTtNwYeUZpyEvcKuXYqlRmkHEL+qkixmj6yFX9jReNcHypO3QOj8StGqu
H/WKwOSnM5Yupv4EblgGPF8ib8tczyxoi+q73sv7iRtQy8wgyAC1gG6T6/qXuY/+1V1K
lyfw==
ARC-Message-Signature: i=2; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20240605;
h=to:from:subject:date:message-id:auto-submitted:sender:reply-to
:mime-version:dkim-signature:dkim-signature:delivered-to;
bh=Hw6hGQbUlNGCz02STp2P+s244T5EBOLlXrfxnO0S/+U=;
fh=mUux9OA2+hLns/mBVi/4Nr5W8MsoxjQs+3G2LAg1TZo=;
b=V+AKgzj8GN/DZNHWPE/MY0blPHHM1Kp85OCTTacCIk/G6dNhx+WmLnIyExrC3i4wmU
i62upyA0a18rhHZRhV1FB4oMMhQVroYLKwh5dFuqFtTARua9DgwYeN6YALL9+rr84n2b
eZe0txkO5dyJgxByumgOymYFgbevrEtd1GWfK2v1BxtQXzqNZ0SKj5PhVCc5WD+toeHu
OEqUuCoHRWpeXYD19OUqv/+MwhPC4t5R5fz8nlPcjxa/fYINuI5+iLhSP7Ki4gzAZFRK
T3zMitsxIv/8zKMrhG5K0cm7Nntn2XBT5zrIDURZW9HEKYLww0yJ8qKXPNL+RtbfGgMq
TiRw==;
darn=xinu.tv
ARC-Authentication-Results: i=2; mx.google.com;
dkim=pass header.i=@google.com header.s=20230601 header.b=bNGW+EgT;
dkim=pass header.i=@gmail.com header.s=20230601 header.b=QfHaLAXu;
spf=pass (google.com: domain of tconvertino@gmail.com designates 209.85.220.73 as permitted sender) smtp.mailfrom=tconvertino@gmail.com;
dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com;
dara=pass header.i=@gmail.com
X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed;
d=1e100.net; s=20230601; t=1768670755; x=1769275555;
h=to:from:subject:date:message-id:auto-submitted:sender:reply-to
:mime-version:dkim-signature:dkim-signature:delivered-to
:x-forwarded-for:x-forwarded-to:x-gm-message-state:from:to:cc
:subject:date:message-id:reply-to;
bh=Hw6hGQbUlNGCz02STp2P+s244T5EBOLlXrfxnO0S/+U=;
b=eTx/TzBvNH8F83uUTAONl5k6vTRn2Id8TE91Mnl0cbJd6GaM9J4DgWVxmBoqfep8nA
hN3H5r01YLOdQNWTWmAV2RUrylBtLMQRqW7xeVnIIUKQNXfavZZKaFpwsuudvjDWGKBo
zesMFCOaEstK+nCpo/bPurb9kprcOh5y/WZjgL7OqtpnnyzhN6DhkhaYbGetIwW4osj5
aCFPOcoMcCYYW8OxEZC6bv2hJzehnV93g8IDY8tBQ88TIf1Kl3uDM8v3oLwMGgmEX+te
PYznnWbJF0vG00cWauIsTnjzUt8SSpnaUXw6PXbHlZxn5Roa/l6hg/tuhs699btYOJm/
izbQ==
X-Forwarded-Encrypted: i=2; AJvYcCUWtYAXLj/f6NFhD1jVOvyY1Jd5fsiQkXHwDFfYixYixyUvud2GXNENdLwj08ultHSVt74PwA==@xinu.tv
X-Gm-Message-State: AOJu0Yx6pI6AZXNq1lGFocBmt39kF0MuPDwo3WPcPrcCg8s1e8EF0iF0
0jOlOq3z4d3WKZqbpCpMIczBtHf5wHUzS1TFiPfYcoHfhnoxLm+dVYhdf0B5b39G2NSwnHIRAcZ
HPVGwj7Cl8dNJOMBLPOevH4CYTDEubbDxDmQOvWE0bhVDk2P+UIU53lYzGkLCnQ==
X-Received: by 2002:a05:6512:4016:b0:59b:7888:62c8 with SMTP id 2adb3069b0e04-59baeed63a1mr2407719e87.33.1768670754691;
Sat, 17 Jan 2026 09:25:54 -0800 (PST)
X-Forwarded-To: gmail@xinu.tv
X-Forwarded-For: couchmoney@gmail.com gmail@xinu.tv
Delivered-To: couchmoney@gmail.com
Received: by 2002:a05:6504:2382:b0:2d3:710a:2457 with SMTP id h2csp2362813lty;
Sat, 17 Jan 2026 09:25:53 -0800 (PST)
X-Received: by 2002:a05:6808:4f0e:b0:450:ac57:48a7 with SMTP id 5614622812f47-45c9c14fc0fmr2557499b6e.59.1768670753179;
Sat, 17 Jan 2026 09:25:53 -0800 (PST)
ARC-Seal: i=1; a=rsa-sha256; t=1768670753; cv=none;
d=google.com; s=arc-20240605;
b=V+T4U8NWyAR1p4yC5XY/I8vxXwtdkLXkIEO6gNBVvJyYi4XbjMMEnoRPAqOULwONFT
7q1V9vArMoZrvS4GNL3dg05tLr0Ug+Frm39+Vp1Wp3UxhQ/yxiby8jhRYkMyaKLZxhR3
2kihw8UgFjdUteHHwKoTDnIkTeKrMKZK8N4bTEzf9LoIXHMZcVaeC5XItuuOUdX6TPXr
xEQKfzCfz3UHY1piusFov9YIr8iBLGnNp6bXJqbRKmnLhOGkt9HQOT9rBl1nmBg5bqQj
4qxTu8Le/CE5qljInXX5iXNYXp1eMD1G6PZ9Hah1hr/wen1VPM6ysNynBlDzlNQUyEMJ
8lNw==
ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20240605;
h=to:from:subject:date:message-id:auto-submitted:sender:reply-to
:mime-version:dkim-signature:dkim-signature;
bh=Hw6hGQbUlNGCz02STp2P+s244T5EBOLlXrfxnO0S/+U=;
fh=mbzrMIWIgWMC0ni1xEx+ViW4J0RLAdLdPT2cX81nTlk=;
b=UhcC2/PG/1T8wb5AzN1QNoaEXR7rs82O/P2CXN7vMVR9JoE3selJIzwIpyWxuKDPK3
GQtEmc8Bqcvcqu//9mWJxsklCkxSXrYnJ0UvykvbmZT7xPhM4r2mpWPluvfxLfEEdbqg
aNgJM1bn4QoYvnjmIF638/SN9dK5TI9seZ04BzbqQxd7Vw5OeccovSpPerSP7ya7l+4k
wOHhvP4mAlB/0bUae8xN/bqS0SIgy+V+cRr3tYEsRb21gJgTT757rHIV0aQu5LSO9t2N
UilB/hh4qvPhaCWmj6I+30ZYD02m9WKPYkwteLA9NXtggMw9WGeywxPZ//pHazzbq7iQ
im+A==;
dara=google.com
ARC-Authentication-Results: i=1; mx.google.com;
dkim=pass header.i=@google.com header.s=20230601 header.b=bNGW+EgT;
dkim=pass header.i=@gmail.com header.s=20230601 header.b=QfHaLAXu;
spf=pass (google.com: domain of tconvertino@gmail.com designates 209.85.220.73 as permitted sender) smtp.mailfrom=tconvertino@gmail.com;
dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com;
dara=pass header.i=@gmail.com
Received: from mail-sor-f73.google.com (mail-sor-f73.google.com. [209.85.220.73])
by mx.google.com with SMTPS id 5614622812f47-45c9e03dfa5sor2738469b6e.10.2026.01.17.09.25.53
for <couchmoney@gmail.com>
(Google Transport Security);
Sat, 17 Jan 2026 09:25:53 -0800 (PST)
Received-SPF: pass (google.com: domain of tconvertino@gmail.com designates 209.85.220.73 as permitted sender) client-ip=209.85.220.73;
Authentication-Results: mx.google.com;
dkim=pass header.i=@google.com header.s=20230601 header.b=bNGW+EgT;
dkim=pass header.i=@gmail.com header.s=20230601 header.b=QfHaLAXu;
spf=pass (google.com: domain of tconvertino@gmail.com designates 209.85.220.73 as permitted sender) smtp.mailfrom=tconvertino@gmail.com;
dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com;
dara=pass header.i=@gmail.com
DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed;
d=google.com; s=20230601; t=1768670752; x=1769275552; dara=google.com;
h=to:from:subject:date:message-id:auto-submitted:sender:reply-to
:mime-version:from:to:cc:subject:date:message-id:reply-to;
bh=Hw6hGQbUlNGCz02STp2P+s244T5EBOLlXrfxnO0S/+U=;
b=bNGW+EgTmg1v7auBVBEmmFGyg6QDqI536axgkCb2SIiknIljcxZLx2KR0hrFlA3lSz
Z89Q1JdMU37Tx8upXXkAQYBe0A42UgQjXEYYfjykMl/PNg7XppVWzevLwkKLmmr/dZ7f
YMcE1DQogEr3RNXJeD92NfJxyOQGskvnzb4rhy22QonzF2UyGy/QX2UtFSz1cZi+35Yq
vTkaernNWU3hf5pAXigHisJTtoJeTRgVNY4ch+gru1X1LmZZzrTgWt6e7hGtsbvlV7cZ
CBM8gqf1LrVLV0Y1PdvS50yack5EFKbyKtmQWAwHBlOABVDwPHbPD9/6N4973C9juedx
HnNg==
DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed;
d=gmail.com; s=20230601; t=1768670753; x=1769275553; dara=google.com;
h=to:from:subject:date:message-id:auto-submitted:sender:reply-to
:mime-version:from:to:cc:subject:date:message-id:reply-to;
bh=Hw6hGQbUlNGCz02STp2P+s244T5EBOLlXrfxnO0S/+U=;
b=QfHaLAXuHdf594PTfEIjd3XOTBUaUaqHXtEArT1QZBEi1Vpf8NyYD6cPKPbjnln3CZ
q5s/sBjI6bmtszfVecPTEv3SEnFgQqQS/PCw3YbMNteemsw4rDNccwV6DSiX/BYMRZIM
4HEgMoLbXrlMlFnjEpWkfb7Kon5Y39C2DNx3sZ3TX/s8fLgYC8JpdUXdZ+LRlr8QzoNH
VLvVwW2iBYOzX9QdBtLdghvnmgvuSxIq1xB0zQNvDixOuG/egq1nDjHna4T75W8qzNEq
+hl8Rng1G2oqWAWUASkwSRvrUvV/NJA3gE+tGD2Isj9d/r4Ppll4jBWOu7KVRPM8Yrld
MYcg==
MIME-Version: 1.0
X-Received: by 2002:a05:6820:1993:b0:65f:67b7:95c2 with SMTP id
006d021491bc7-661179f382fmr2914920eaf.55.1768670752809; Sat, 17 Jan 2026
09:25:52 -0800 (PST)
Reply-To: tconvertino@gmail.com
Sender: Google Calendar <calendar-notification@google.com>
Auto-Submitted: auto-generated
Message-ID: <calendar-c95e83d6-7062-41ea-8c57-5b7bc2a79c76@google.com>
Date: Sat, 17 Jan 2026 17:25:52 +0000
Subject: New event: Emery Sleeps Over @ Sun Jan 18, 2026 (tconvertino@gmail.com)
From: tconvertino@gmail.com
To: couchmoney@gmail.com
Content-Type: multipart/alternative; boundary="000000000000f22916064898bf45"
X-Rspamd-Queue-Id: B744880023
X-Rspamd-Server: phx
X-Spamd-Result: default: False [-0.90 / 15.00];
URI_COUNT_ODD(1.00)[1];
ARC_ALLOW(-1.00)[google.com:s=arc-20240605:i=2];
DMARC_POLICY_ALLOW(-0.50)[gmail.com,none];
R_DKIM_ALLOW(-0.20)[google.com:s=20230601,gmail.com:s=20230601];
R_SPF_ALLOW(-0.20)[+ip4:209.85.128.0/17];
MANY_INVISIBLE_PARTS(0.10)[2];
MIME_GOOD(-0.10)[multipart/alternative,text/plain];
FREEMAIL_TO(0.00)[gmail.com];
RCVD_COUNT_THREE(0.00)[3];
RCVD_TLS_LAST(0.00)[];
FORGED_SENDER(0.00)[tconvertino@gmail.com,couchmoney@gmail.com];
RCPT_COUNT_ONE(0.00)[1];
RCVD_IN_DNSWL_NONE(0.00)[209.85.220.73:received];
TAGGED_FROM(0.00)[caf_=gmail=xinutv];
FREEMAIL_REPLYTO(0.00)[gmail.com];
MIME_TRACE(0.00)[0:+,1:+,2:~];
FREEMAIL_FROM(0.00)[gmail.com];
FROM_NEQ_ENVFROM(0.00)[tconvertino@gmail.com,couchmoney@gmail.com];
MISSING_XM_UA(0.00)[];
HAS_REPLYTO(0.00)[tconvertino@gmail.com];
DNSWL_BLOCKED(0.00)[209.85.167.48:from];
DWL_DNSWL_NONE(0.00)[gmail.com:dkim];
TO_DN_NONE(0.00)[];
FREEMAIL_ENVFROM(0.00)[gmail.com];
FORGED_SENDER_FORWARDING(0.00)[];
DKIM_TRACE(0.00)[google.com:+,gmail.com:+];
DWL_DNSWL_BLOCKED(0.00)[google.com:dkim];
TO_DOM_EQ_FROM_DOM(0.00)[];
FROM_NO_DN(0.00)[];
FWD_GOOGLE(0.00)[couchmoney@gmail.com];
ASN(0.00)[asn:15169, ipnet:209.85.128.0/17, country:US];
RWL_MAILSPIKE_POSSIBLE(0.00)[209.85.167.48:from];
REPLYTO_EQ_FROM(0.00)[]
X-Rspamd-Action: no action
X-TUID: GNj+V6W3PxE3
--000000000000f22916064898bf45
Content-Type: text/plain; charset="UTF-8"; format=flowed; delsp=yes
Emery Sleeps Over
Sunday Jan 18, 2026
Organizer
tconvertino@gmail.com
tconvertino@gmail.com
~~//~~
Invitation from Google Calendar: https://calendar.google.com/calendar/
You are receiving this email because you are subscribed to calendar
notifications. To stop receiving these emails, go to
https://calendar.google.com/calendar/r/settings, select this calendar, and
change "Other notifications".
Forwarding this invitation could allow any recipient to send a response to
the organizer, be added to the guest list, invite others regardless of
their own invitation status, or modify your RSVP.
Learn more https://support.google.com/calendar/answer/37135#forwarding
--000000000000f22916064898bf45
Content-Type: text/html; charset="UTF-8"
Content-Transfer-Encoding: quoted-printable
<!doctype html><html xmlns=3D"http://www.w3.org/1999/xhtml" xmlns:v=3D"urn:=
schemas-microsoft-com:vml" xmlns:o=3D"urn:schemas-microsoft-com:office:offi=
ce"><head><title></title><!--[if !mso]><meta http-equiv=3D"X-UA-Compatible"=
content=3D"IE=3Dedge"><![endif]--><meta http-equiv=3D"Content-Type" conten=
t=3D"text/html; charset=3DUTF-8"><meta name=3D"viewport" content=3D"width=
=3Ddevice-width,initial-scale=3D1"><meta name=3D"color-scheme" content=3D"l=
ight dark"><meta name=3D"supported-color-schemes" content=3D"light dark">
<style>
body, html {
font-family: Roboto, Helvetica, Arial, sans-serif;
}
body {
margin: 0;
padding: 0;
-webkit-font-smoothing: antialiased;
-webkit-text-size-adjust: 100%;
-ms-text-size-adjust: 100%;
}
#outlook a {
padding: 0;
}
.ReadMsgBody {
width: 100%;
}
.ExternalClass {
width: 100%;
}
.ExternalClass * {
line-height: 100%;
}
table,
td {
mso-table-lspace: 0pt;
mso-table-rspace: 0pt;
}
img {
border: 0;
height: auto;
line-height: 100%;
outline: none;
text-decoration: none;
-ms-interpolation-mode: bicubic;
}
p {
display: block;
margin: 13px 0;
}
</style>
<!--[if !mso]><!-->
<style>
@media only screen and (max-width:580px) {
@-ms-viewport {
width: 320px;
}
@viewport {
width: 320px;
}
}
</style>
<!--<![endif]-->
<!--[if mso]>
<xml>
<o:OfficeDocumentSettings>
<o:AllowPNG/>
<o:PixelsPerInch>96</o:PixelsPerInch>
</o:OfficeDocumentSettings>
</xml>
<![endif]-->
<!--[if lte mso 11]>
<style>
.outlook-group-fix { width:100% !important; }
</style>
<![endif]-->
<!--[if !mso]><!-- -->
<style>body, html {font-family:Roboto,Helvetica,Arial,sans-serif;}@font-f=
ace {
font-family: 'Roboto';
font-style: normal;
font-weight: 400;
font-stretch: normal;
src: url(//fonts.gstatic.com/s/roboto/v48/KFOMCnqEu92Fr1ME7kSn66aGLdTylUA=
MQXC89YmC2DPNWubEbVmUiA8.ttf) format('truetype');
}
@font-face {
font-family: 'Roboto';
font-style: normal;
font-weight: 500;
font-stretch: normal;
src: url(//fonts.gstatic.com/s/roboto/v48/KFOMCnqEu92Fr1ME7kSn66aGLdTylUA=
MQXC89YmC2DPNWub2bVmUiA8.ttf) format('truetype');
}
@font-face {
font-family: 'Roboto';
font-style: normal;
font-weight: 700;
font-stretch: normal;
src: url(//fonts.gstatic.com/s/roboto/v48/KFOMCnqEu92Fr1ME7kSn66aGLdTylUA=
MQXC89YmC2DPNWuYjalmUiA8.ttf) format('truetype');
}
@font-face {
font-family: 'Material Icons Extended';
font-style: normal;
font-weight: 400;
src: url(//fonts.gstatic.com/s/materialiconsextended/v154/kJEjBvgX7BgnkSr=
UwT8UnLVc38YydejYY-oE_LvM.ttf) format('truetype');
}
@font-face {
font-family: 'Google Material Icons';
font-style: normal;
font-weight: 400;
src: url(//fonts.gstatic.com/s/googlematerialicons/v144/Gw6kwdfw6UnXLJCcm=
afZyFRXb3BL9rvi0QZG3g.otf) format('opentype');
}
.google-material-icons {
font-family: 'Google Material Icons';
font-weight: normal;
font-style: normal;
font-size: 24px;
line-height: 1;
letter-spacing: normal;
text-transform: none;
display: inline-block;
white-space: nowrap;
word-wrap: normal;
direction: ltr;
}
@font-face {
font-family: 'Google Material Icons Filled';
font-style: normal;
font-weight: 400;
src: url(//fonts.gstatic.com/s/googlematerialiconsfilled/v118/WWXFlimHYg6=
HKI3TavMkbKdhBmDvgach8TVpeGsuueSZJH4.otf) format('opentype');
}
.google-material-icons-filled {
font-family: 'Google Material Icons Filled';
font-weight: normal;
font-style: normal;
font-size: 24px;
line-height: 1;
letter-spacing: normal;
text-transform: none;
display: inline-block;
white-space: nowrap;
word-wrap: normal;
direction: ltr;
}
@font-face {
font-family: 'Google Sans';
font-style: normal;
font-weight: 400;
src: url(//fonts.gstatic.com/s/googlesans/v14/4UaGrENHsxJlGDuGo1OIlL3Owps=
.ttf) format('truetype');
}
@font-face {
font-family: 'Google Sans';
font-style: normal;
font-weight: 500;
src: url(//fonts.gstatic.com/s/googlesans/v14/4UabrENHsxJlGDuGo1OIlLU94Yt=
zCwM.ttf) format('truetype');
}
@font-face {
font-family: 'Google Sans';
font-style: normal;
font-weight: 700;
src: url(//fonts.gstatic.com/s/googlesans/v14/4UabrENHsxJlGDuGo1OIlLV154t=
zCwM.ttf) format('truetype');
}
</style><!--<![endif]-->
<style>
.body-container {
padding-left: 16px;
padding-right: 16px;
}
</style>
=20
<style>
u+.body .body-container,
body[data-outlook-cycle] .body-container,
#MessageViewBody .body-container {
padding-left: 0;
padding-right: 0;
}
</style>
=20
<style>
@media only screen and (min-width:580px) {
.column-per-37 {
width: 37% !important;
max-width: 37%;
}
.column-per-63 {
width: 63% !important;
max-width: 63%;
}
}
</style>
=20
<style>
.appointment-buttons th {
display: block;
clear: both;
float: left;
margin-top: 12px;
}
.appointment-buttons th a {
float: left;
}
#MessageViewBody .appointment-buttons th {
margin-top: 24px;
}
</style>
=20
<style>
@media only screen and (max-width:580px) {
table.full-width-mobile {
width: 100% !important;
}
td.full-width-mobile {
width: auto !important;
}
}
</style>
<style>
.main-container-inner,
.info-bar-inner {
padding: 12px 16px !important;
}
.main-column-table-ltr {
padding-right: 0 !important;
}
.main-column-table-rtl {
padding-left: 0 !important;
}
@media only screen and (min-width:580px) {
.main-container-inner {
padding: 24px 32px !important;
}
.info-bar-inner {
padding: 12px 32px !important;
}
.main-column-table-ltr {
padding-right: 32px !important;
}
.main-column-table-rtl {
padding-left: 32px !important;
}
.appointment-buttons th {
display: table-cell;
clear: none;
}
}
.primary-text {
color: #3c4043 !important;
}
.secondary-text,
.phone-number a {
color: #70757a !important;
}
.accent-text {
color: #1a73e8 !important;
}
.accent-text-dark {
color: #185abc !important;
}
.grey-button-text,
.attachment-chip a {
color: #5f6368 !important;
}
.primary-button {
background-color: #1a73e8 !important;
}
.primary-button-text {
color: #fff !important;
}
.underline-on-hover:hover {
text-decoration: underline !important;
}
.grey-infobar-text {
color: #202124 !important;
}
@media (prefers-color-scheme: dark) {
.primary-text:not([class^=3D"x_"]) {
color: #e8eaed !important;
}
.secondary-text:not([class^=3D"x_"]),
.phone-number:not([class^=3D"x_"]) a {
color: #9aa0a6 !important;
}
.grey-button-text:not([class^=3D"x_"]),
.attachment-chip:not([class^=3D"x_"]) a {
color: #bdc1c6 !important;
}
.accent-text:not([class^=3D"x_"]),
.hairline-button-text:not([class^=3D"x_"]) {
color: #8ab4f8 !important;
}
.primary-button:not([class^=3D"x_"]) {
background-color: #8ab4f8 !important;
}
.primary-button-text:not([class^=3D"x_"]) {
color: #202124 !important;
}
}
</style>
<style>
@media (prefers-color-scheme: dark) {
.cse-banner:not([class^=3D"x_"]) {
background-color: #3c4043 !important; /* Google Grey 800 */
}
.encryption-icon:not([class^=3D"x_"]) {
/* WARNING: This causes the whole style tag to get stripped in Gm=
ail. */
background-image: url('https://fonts.gstatic.com/s/i/googlemateri=
aliconsfilled/encrypted/v3/gm_grey200-24dp/2x/gm_filled_encrypted_gm_grey20=
0_24dp.png') !important;
}
}
</style>
<!--[if !mso]><!-->
<style>
.prevent-link a {
color: inherit !important;
text-decoration: none !important;
font-size: inherit !important;
font-family: inherit !important;
font-weight: inherit !important;
line-height: inherit !important;
}
</style>
<!--<![endif]-->
<!--[if mso | IE]>
<style>
.main-container-inner {
padding: 24px 32px !important;
}
.info-bar-inner {
padding: 12px 32px !important;
}
.cse-banner .encryption-icon {
/* We use the IE workaround instead. */
background-image: none !important;
}
.cse-banner .encryption-icon .ms-fallback {
display: block !important;
}
/* NB: Some MS clients ignore dark-scheme styling and apply their o=
wn, so there's nothing we can do to help there. */
@media (prefers-color-scheme: dark) {
.cse-banner:not([class^=3D"x_"]) .encryption-icon .ms-fallback {
display: none !important;
}
.cse-banner:not([class^=3D"x_"]) .encryption-icon .ms-fallback-da=
rk {
display: block !important;
}
}
</style>
<![endif]-->
</head><body class=3D"body"><span itemscope itemtype=3D"http://schema.org=
/InformAction"><span style=3D"display:none" itemprop=3D"about" itemscope it=
emtype=3D"http://schema.org/Person"><meta itemprop=3D"description" content=
=3D"Invitation from tconvertino@gmail.com"/></span><span itemprop=3D"object=
" itemscope itemtype=3D"http://schema.org/Event"><meta itemprop=3D"eventSta=
tus" content=3D"http://schema.org/EventScheduled"/><span itemprop=3D"publis=
her" itemscope itemtype=3D"http://schema.org/Organization"><meta itemprop=
=3D"name" content=3D"Google Calendar"/></span><meta itemprop=3D"eventId/goo=
gleCalendar" content=3D"7bsl09rb144q35feflbl0q2v58"/><span style=3D"display=
: none; font-size: 1px; color: #fff; line-height: 1px; height: 0; max-heigh=
t: 0; width: 0; max-width: 0; opacity: 0; overflow: hidden;" itemprop=3D"na=
me">Emery Sleeps Over</span><meta itemprop=3D"url" content=3D"https://calen=
dar.google.com/calendar/r?eid=3DN2JzbDA5cmIxNDRxMzVmZWZsYmwwcTJ2NTggdGNvbnZ=
lcnRpbm9AbQ&amp;es=3D1"/><span aria-hidden=3D"true"><time itemprop=3D"start=
Date" datetime=3D"20260118"></time><time itemprop=3D"endDate" datetime=3D"2=
0260119"></time></span><div style=3D"display: none; font-size: 1px; color: =
#fff; line-height: 1px; height: 0; max-height: 0; width: 0; max-width: 0; o=
pacity: 0; overflow: hidden;">You have been invited by tconvertino@gmail.co=
m to attend an event named Emery Sleeps Over on Sunday Jan 18, 2026.</div><=
table border=3D"0" cellpadding=3D"0" cellspacing=3D"0" role=3D"presentation=
" align=3D"center" style=3D"width:100%;" class=3D"body-container"><tbody><t=
r><td style=3D"" class=3D"" align=3D"left"><!--[if mso | IE]><table border=
=3D"0" cellpadding=3D"0" cellspacing=3D"0" role=3D"presentation"><tr><td he=
ight=3D"16" style=3D"height:16px;"><![endif]--><div style=3D"height:16px;" =
aria-hidden=3D"true"> &nbsp; </div><!--[if mso | IE]></td></tr></table><![e=
ndif]--><table border=3D"0" cellpadding=3D"0" cellspacing=3D"0" role=3D"pre=
sentation" align=3D"center" style=3D"width:100%;" class=3D""><tbody><tr><td=
style=3D"border: solid 1px #dadce0; border-radius: 8px; direction: rtl; fo=
nt-size: 0; padding: 24px 32px; text-align: left; vertical-align: top;" cla=
ss=3D"main-container-inner"><!--[if mso | IE]><table border=3D"0" cellpaddi=
ng=3D"0" cellspacing=3D"0" role=3D"presentation"><tr><![endif]--><div class=
=3D"" style=3D"font-size: 13px; text-align: left; direction: ltr; display: =
inline-block; vertical-align: top; width: 100%;overflow: hidden; word-wrap:=
break-word;"><table border=3D"0" cellpadding=3D"0" cellspacing=3D"0" role=
=3D"presentation" width=3D"100%" class=3D"main-column-table-ltr" style=3D"p=
adding-right: 32px; padding-left: 0;;table-layout: fixed;"><tbody><tr><td c=
lass=3D"main-column-td" style=3D"padding:0; vertical-align:top;"><table bor=
der=3D"0" cellpadding=3D"0" cellspacing=3D"0" role=3D"presentation" width=
=3D"100%" style=3D"table-layout: fixed;"><tr><td style=3D"font-size: 0; pad=
ding: 0; text-align: left; word-break: break-word;;padding-bottom:24px;"><d=
iv style=3D"font-family: Roboto, sans-serif;font-style: normal; font-weight=
: 400; font-size: 14px; line-height: 20px; letter-spacing: 0.2px;color: #3c=
4043; text-decoration: none;" class=3D"primary-text" role=3D"presentation">=
<span aria-hidden=3D"true"><time itemprop=3D"startDate" datetime=3D"2026011=
8"></time><time itemprop=3D"endDate" datetime=3D"20260119"></time></span><t=
able border=3D"0" cellpadding=3D"0" cellspacing=3D"0" role=3D"presentation"=
style=3D"padding-bottom: 4px;"><tr><td><h2 class=3D"primary-text" style=3D=
"font-size: 14px;color: #3c4043; text-decoration: none;font-weight: 700;-we=
bkit-font-smoothing: antialiased;margin: 0; padding: 0;">When</h2></td></tr=
></table><span>Sunday Jan 18, 2026</span></div></td></tr><tr><td style=3D"f=
ont-size: 0; padding: 0; text-align: left; word-break: break-word;;padding-=
bottom:24px;"><div style=3D"font-family: Roboto, sans-serif;font-style: nor=
mal; font-weight: 400; font-size: 14px; line-height: 20px; letter-spacing: =
0.2px;color: #3c4043; text-decoration: none;" class=3D"primary-text" role=
=3D"presentation"><table border=3D"0" cellpadding=3D"0" cellspacing=3D"0" r=
ole=3D"presentation" style=3D"padding-bottom: 4px;"><tr><td><h2 class=3D"pr=
imary-text" style=3D"font-size: 14px;color: #3c4043; text-decoration: none;=
font-weight: 700;-webkit-font-smoothing: antialiased;margin: 0; padding: 0;=
">Calendar</h2></td></tr></table>tconvertino@gmail.com</div></td></tr><tr><=
td style=3D"font-size: 0; padding: 0; text-align: left; word-break: break-w=
ord;;padding-bottom:24px;"><div style=3D"font-family: Roboto, sans-serif;fo=
nt-style: normal; font-weight: 400; font-size: 14px; line-height: 20px; let=
ter-spacing: 0.2px;color: #3c4043; text-decoration: none;" class=3D"primary=
-text" role=3D"presentation"><table border=3D"0" cellpadding=3D"0" cellspac=
ing=3D"0" role=3D"presentation" style=3D"padding-bottom: 4px;"><tr><td><h2 =
class=3D"primary-text" style=3D"font-size: 14px;color: #3c4043; text-decora=
tion: none;font-weight: 700;-webkit-font-smoothing: antialiased;margin: 0; =
padding: 0;">Organizer</h2></td></tr></table><div style=3D"color: #3c4042;"=
><span class=3D"notranslate"><a class=3D"primary-text underline-on-hover" s=
tyle=3D"display: inline-block;;color: #3c4043; text-decoration: none;" href=
=3D"mailto:tconvertino@gmail.com">tconvertino@gmail.com</a></span><span ite=
mprop=3D"organizer" itemscope itemtype=3D"http://schema.org/Person"><meta i=
temprop=3D"name" content=3D"tconvertino@gmail.com"/><meta itemprop=3D"email=
" content=3D"tconvertino@gmail.com"/></span></div></div></td></tr><tr><td s=
tyle=3D"font-size: 0; padding: 0; text-align: left; word-break: break-word;=
;padding-bottom:24px;"><div style=3D"font-family: Roboto, sans-serif;font-s=
tyle: normal; font-weight: 400; font-size: 14px; line-height: 20px; letter-=
spacing: 0.2px;color: #3c4043; text-decoration: none;" class=3D"primary-tex=
t" role=3D"presentation"><table border=3D"0" cellpadding=3D"0" cellspacing=
=3D"0" role=3D"presentation" style=3D"padding-bottom: 4px;"><tr><td><h2 cla=
ss=3D"primary-text" style=3D"font-size: 14px;color: #3c4043; text-decoratio=
n: none;font-weight: 700;-webkit-font-smoothing: antialiased;margin: 0; pad=
ding: 0;">Guests</h2></td></tr></table><div style=3D"padding-bottom: 4px; t=
ext-align: left;;color: #3c4042;"></div><a href=3D"https://calendar.google.=
com/calendar/r?eid=3DN2JzbDA5cmIxNDRxMzVmZWZsYmwwcTJ2NTggdGNvbnZlcnRpbm9AbQ=
&amp;es=3D1" style=3D"display: inline-block;;color: #1a73e8; text-decoratio=
n: none;font-weight: 700;" target=3D"_blank" class=3D"accent-text underline=
-on-hover">View all guest info</a></div></td></tr></table></td></tr></tbody=
></table></div><!--[if mso | IE]></tr></table><![endif]--></td></tr></tbody=
></table><table border=3D"0" cellpadding=3D"0" cellspacing=3D"0" role=3D"pr=
esentation" align=3D"center" style=3D"width:100%;" class=3D""><tbody><tr><t=
d style=3D"font-size: 0; padding: 0; text-align: left; word-break: break-wo=
rd;;padding:4px 12px;" class=3D"" align=3D"left"><div class=3D"secondary-te=
xt" style=3D"color: #70757a; text-decoration: none;font-family: Roboto, san=
s-serif;font-size: 12px; line-height: 16px; mso-line-height-rule: exactly; =
text-align: left;"><p>Invitation from <a href=3D"https://calendar.google.co=
m/calendar/" class=3D"accent-text underline-on-hover" style=3D"font-family:=
Roboto, sans-serif;font-size: 12px; line-height: 16px; mso-line-height-rul=
e: exactly;;color: #1a73e8; text-decoration: none;" target=3D"_blank">Googl=
e Calendar</a></p><p>You are receiving this email because you are subscribe=
d to calendar notifications. To stop receiving these emails, go to <a href=
=3D"https://calendar.google.com/calendar/r/settings" class=3D"accent-text u=
nderline-on-hover" style=3D"font-family: Roboto, sans-serif;font-size: 12px=
; line-height: 16px; mso-line-height-rule: exactly;;color: #1a73e8; text-de=
coration: none;" target=3D"_blank">Calendar settings</a>, select this calen=
dar, and change "Other notifications".</p><p>Forwarding this invitation cou=
ld allow any recipient to send a response to the organizer, be added to the=
guest list, invite others regardless of their own invitation status, or mo=
dify your RSVP. <a class=3D"accent-text underline-on-hover" style=3D"font-f=
amily: Roboto, sans-serif;font-size: 12px; line-height: 16px; mso-line-heig=
ht-rule: exactly;;color: #1a73e8; text-decoration: none;" href=3D"https://s=
upport.google.com/calendar/answer/37135#forwarding">Learn more</a></p></div=
></td></tr></tbody></table></td></tr></tbody></table></span></span></body><=
/html>
--000000000000f22916064898bf45--

View File

@@ -0,0 +1,733 @@
Return-Path: <couchmoney+caf_=gmail=xinu.tv@gmail.com>
Delivered-To: bill@xinu.tv
Received: from phx.xinu.tv [74.207.253.222]
by nixos-01.h.xinu.tv with IMAP (fetchmail-6.5.6)
for <wathiede@localhost> (single-drop); Sat, 17 Jan 2026 07:15:51 -0800 (PST)
Received: from phx.xinu.tv
by phx.xinu.tv with LMTP
id EBhFKaana2lD2BMAJR8clQ
(envelope-from <couchmoney+caf_=gmail=xinu.tv@gmail.com>)
for <bill@xinu.tv>; Sat, 17 Jan 2026 07:15:50 -0800
X-Original-To: gmail@xinu.tv
Received: from mail-lf1-f47.google.com (mail-lf1-f47.google.com [209.85.167.47])
by phx.xinu.tv (Postfix) with ESMTPS id 9ED9E80023
for <gmail@xinu.tv>; Sat, 17 Jan 2026 07:15:49 -0800 (PST)
Received: by mail-lf1-f47.google.com with SMTP id 2adb3069b0e04-59b72a1e2f0so3298670e87.0
for <gmail@xinu.tv>; Sat, 17 Jan 2026 07:15:49 -0800 (PST)
ARC-Seal: i=2; a=rsa-sha256; t=1768662948; cv=pass;
d=google.com; s=arc-20240605;
b=P8ukIhKVppSoOtIG2772uR9xtbgbpQjTdNpTjI3WHjWj7WKqgmV4ndotO/y7GwqvOX
OwYJbwY9cIR/PPOGf0/UfUSfPTp6iWwfapN3B5YDoSrOk3ef8e0HXRcSMeIG9D0JWfKB
YTFLsg6dqxdrPLsyDhFeJQQFK4+qb4gndM7SEw9UZtvvEbzGmGro9QFcO+uXsPZ6jQsb
UNn7VJhOqi3sc+mNAmPSNN1KSjeZ5wQoJDKHAhc39P7HlCI6Upm3UgyejjToTNa5pFUH
YfqgrPEU0NsJWZNUfnItYhChHXZyHE9iUcK8eDO7qRyvVAwkEdxmc34EC4hOJsHxdLli
ik4A==
ARC-Message-Signature: i=2; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20240605;
h=to:from:subject:date:message-id:auto-submitted:sender:reply-to
:mime-version:dkim-signature:dkim-signature:delivered-to;
bh=DSYDWRwqF5i+Yosvco75LpPB3hH2JqH2JZElsHalDcM=;
fh=opiyoKXgWpm8kcAa3Nw4m5S9DBRIdfMLcfsQwVK5zKs=;
b=hyfgpNzlhW6aPYNzOd5CkaqQ2qYr/Sb25Qti1ez7y+n0SxJM/QbpyMCafAFv1s6vSR
jdbmeaofRjmDkL0uDdoQb0E5TnycNKP3r4FWEKc2KZkfnKEtWWOivFDIvQDPF4NpYNn3
9SFZkfmVa2TSfrnH4wRG37gCQgc6gzDpl6a2oZRmM4T44xbIXCIseFSalR+oUTFCJU3M
XSBCNjV7w/V8S2hiH72Ace8enjUG+DRbeE8aYiNwu0XkECojPxGgBuM9OJPSmYzNt3LW
7M+weqkx0Byoi6UDaI2NQqXH4wkSrjmIFcX+84jglfl2zOsPGwgehJYdo/QcPMXtQohY
sMZA==;
darn=xinu.tv
ARC-Authentication-Results: i=2; mx.google.com;
dkim=pass header.i=@google.com header.s=20230601 header.b=UV12BjMR;
dkim=pass header.i=@gmail.com header.s=20230601 header.b=HTQ+XWtf;
spf=pass (google.com: domain of tconvertino@gmail.com designates 209.85.220.73 as permitted sender) smtp.mailfrom=tconvertino@gmail.com;
dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com;
dara=pass header.i=@gmail.com
X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed;
d=1e100.net; s=20230601; t=1768662948; x=1769267748;
h=to:from:subject:date:message-id:auto-submitted:sender:reply-to
:mime-version:dkim-signature:dkim-signature:delivered-to
:x-forwarded-for:x-forwarded-to:x-gm-message-state:from:to:cc
:subject:date:message-id:reply-to;
bh=DSYDWRwqF5i+Yosvco75LpPB3hH2JqH2JZElsHalDcM=;
b=wCE8iOXwi+1PwnSvMIbzsg5Pdo1jXyrqQjqZN3UN9bXyLJmoKV+a5l4cE7La2TkWIe
zdw/qcgetdMWPrJncmnLYptNMOMhHf8OmgDbtKbZLo1WVnY6OcnTTCWljhm9l3p1sNCJ
NNvrE79CNsi4mB30/gh5GUSFldNs4CKXLoX5U1JyaR0URBNSdgTtWHQEk9Zip6lzSJ/W
3myHdqXjlFIBn8JdN2F+Uvpe4SOcpUYivDzRzGgKBVBybgL+7/B2FVxu1DSqImvvdnew
G45//ZKZLk1NZTwB9b3fVu1ZRrtVwXykyTsLWQs/L+h2RatzY5cbRknOUEg+Sh12glbJ
Mvzg==
X-Forwarded-Encrypted: i=2; AJvYcCWTlZh7K3d3elcvXPqK9OnJxLWgTqppYJqLWqVOMFJDYDcmSN8i4AmT2kLKruApNvLk5mchcQ==@xinu.tv
X-Gm-Message-State: AOJu0YzRCKajwo80HG2ZJYkIh92JNKbnNkbFqaG5Dye25AAdsFY1p/Sa
HUpoeDQQxTSXezOEygAE9wR8sFTjlB4uRgVqtHJ5s0ml5G9sQVG9Ir31aJU5vBs4OpeeAa01KHq
OqoyqsCDybPJBW43d/qWOxpcpIORkUKMmPkNGNzU1+hFlox6fAt4TasNy8lbiLw==
X-Received: by 2002:a05:6512:2507:b0:59b:b55a:a293 with SMTP id 2adb3069b0e04-59bb55aa3b5mr1328824e87.34.1768662947779;
Sat, 17 Jan 2026 07:15:47 -0800 (PST)
X-Forwarded-To: gmail@xinu.tv
X-Forwarded-For: couchmoney@gmail.com gmail@xinu.tv
Delivered-To: couchmoney@gmail.com
Received: by 2002:a05:6504:2382:b0:2d3:710a:2457 with SMTP id h2csp2302771lty;
Sat, 17 Jan 2026 07:15:46 -0800 (PST)
X-Received: by 2002:a05:6808:221e:b0:45a:6ef9:79 with SMTP id 5614622812f47-45c9d86c3acmr2292013b6e.52.1768662946398;
Sat, 17 Jan 2026 07:15:46 -0800 (PST)
ARC-Seal: i=1; a=rsa-sha256; t=1768662946; cv=none;
d=google.com; s=arc-20240605;
b=KdiLKIN7OQmPF5O3bozzB9t0JL/To2npnCMuMVg4Iawwtni3NbkM+Y4nXB3Wtm5kAj
Sa6HsrnbgNz3D8lp3nJB8bXexvVTc73ZQFDKtKnDRj7sv/8eK0qMpCkRb3Rhgu2cQEfa
lYv7E0pEj0qcMxDM5osGhdhrwsV8BnEboHL38bCfhDxSmcmGsmPgiC5IWiyhcWq8G2W6
XwwPDdcHXk4wLwy4AQ1/NR5q130ELNP1f2e1Xq+xxLhfuV7GzhgOCTxYBIzrzJdM0kC/
Sxd1ejz+WLbowjVP2s4rStRtCvHU/g1LlPGmdnmyN9wQ+Tkya+Q7LjRWWB99e04qENOE
3E9g==
ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20240605;
h=to:from:subject:date:message-id:auto-submitted:sender:reply-to
:mime-version:dkim-signature:dkim-signature;
bh=DSYDWRwqF5i+Yosvco75LpPB3hH2JqH2JZElsHalDcM=;
fh=mbzrMIWIgWMC0ni1xEx+ViW4J0RLAdLdPT2cX81nTlk=;
b=R24D+rdvNBJO+lAlMBX/5KkpnryHlsj/t5x1rJvWsKG/yrKJe81Epa9aDOWh5pljV4
Cg7IH6KrYLzYYMNQ83stCE7zIBs7sjtBl81bd/OpWCoXgJEAnE2bYs958fk3z8jINzBv
huNTL3ijCNiZeI/y5Ye/S8avkRfWf1tNn7nNc7oVf6Xc+ujGIk28KL3EGerj29au519E
MqL6BVYlSAx9VcBXY2vRZft2xDJo+JiKIw2n+XJT7+Ax0LubBX5BCfcRinBFT/XsVFTD
zxNaxxq9DbbAtr78NgacXMOxmo8bJYi/MD0wo0BVZBxd3kXhKF367iEKBreXu7tYPp5i
d7tg==;
dara=google.com
ARC-Authentication-Results: i=1; mx.google.com;
dkim=pass header.i=@google.com header.s=20230601 header.b=UV12BjMR;
dkim=pass header.i=@gmail.com header.s=20230601 header.b=HTQ+XWtf;
spf=pass (google.com: domain of tconvertino@gmail.com designates 209.85.220.73 as permitted sender) smtp.mailfrom=tconvertino@gmail.com;
dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com;
dara=pass header.i=@gmail.com
Received: from mail-sor-f73.google.com (mail-sor-f73.google.com. [209.85.220.73])
by mx.google.com with SMTPS id 5614622812f47-45ca0e0d20dsor1067583b6e.18.2026.01.17.07.15.46
for <couchmoney@gmail.com>
(Google Transport Security);
Sat, 17 Jan 2026 07:15:46 -0800 (PST)
Received-SPF: pass (google.com: domain of tconvertino@gmail.com designates 209.85.220.73 as permitted sender) client-ip=209.85.220.73;
Authentication-Results: mx.google.com;
dkim=pass header.i=@google.com header.s=20230601 header.b=UV12BjMR;
dkim=pass header.i=@gmail.com header.s=20230601 header.b=HTQ+XWtf;
spf=pass (google.com: domain of tconvertino@gmail.com designates 209.85.220.73 as permitted sender) smtp.mailfrom=tconvertino@gmail.com;
dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com;
dara=pass header.i=@gmail.com
DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed;
d=google.com; s=20230601; t=1768662946; x=1769267746; dara=google.com;
h=to:from:subject:date:message-id:auto-submitted:sender:reply-to
:mime-version:from:to:cc:subject:date:message-id:reply-to;
bh=DSYDWRwqF5i+Yosvco75LpPB3hH2JqH2JZElsHalDcM=;
b=UV12BjMRpTIHG34HTOnEB4ApBu4DnXlft6fU+aIV2TzL1HFmpDf9rz1riEoHyaWSGU
P7ydXwhdNhl1b9SVRu+jhtsd0Wt7vLGTb/ru1OZEnxC8P2A9jCdqjT3J9GwtpZxcv4nh
ipdJF6LYZ6yt1H2AftMJf59/9L1HbiwMs3MceHqj+R0AlU4KNtuevB/ImTump5XiSH/8
iDgvVfqyvm2kYwp9yV4cMXWUIQG6bWHB/No01kLG2PtgOy45chmfp/P22/ZyY3wEFd/U
044fReZdGmqwqGfhM81y2+WVwuzQltjv8dZ2yf1SeSVOLmrc8Uc5lYFvzQJK7NiaVsyv
sYWQ==
DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed;
d=gmail.com; s=20230601; t=1768662946; x=1769267746; dara=google.com;
h=to:from:subject:date:message-id:auto-submitted:sender:reply-to
:mime-version:from:to:cc:subject:date:message-id:reply-to;
bh=DSYDWRwqF5i+Yosvco75LpPB3hH2JqH2JZElsHalDcM=;
b=HTQ+XWtfciYuHpwOHCqsF6ctCQc5eRWblgRxVdEfjGFYy2TmgSb/jA5+HldQ7hxk5S
mfppkAhF82R1zQM96e2QMnoamqslIpum/+pvdIfxbvXiAytCNOmQZoc1xhlTz5CCkpni
g23H0O3kiXSdzhG4uLJgsGMaNR/rAkslbMW7ARNTluvuWM60d9At9ZCazZCVLq2C0nhS
0HvaBa5uxi19A6l/55NmEbhPvAHXSw7V9SEUF8TTBuGLfZEkrjUF+RxfAgN/+NuZR5o7
yOAlNI/iEi+F8T7Is+ZmjbHYysZu01/nzldZnZEQamYuUQBeKpbNHSEGbVrpD3+AuT6f
/t8w==
MIME-Version: 1.0
X-Received: by 2002:a05:6820:1ca1:b0:65f:7009:d6e5 with SMTP id
006d021491bc7-661188b9d4bmr1995718eaf.22.1768662945924; Sat, 17 Jan 2026
07:15:45 -0800 (PST)
Reply-To: tconvertino@gmail.com
Sender: Google Calendar <calendar-notification@google.com>
Auto-Submitted: auto-generated
Message-ID: <calendar-3c0029f2-74ee-41a1-b1f0-0716665ce0d3@google.com>
Date: Sat, 17 Jan 2026 15:15:45 +0000
Subject: Updated invitation: painting class @ Thu Feb 12, 2026 7pm - 9pm (PST) (tconvertino@gmail.com)
From: tconvertino@gmail.com
To: couchmoney@gmail.com
Content-Type: multipart/alternative; boundary="0000000000009e8d3e064896eeed"
X-Rspamd-Queue-Id: 9ED9E80023
X-Rspamd-Server: phx
X-Spamd-Result: default: False [-0.70 / 15.00];
URI_COUNT_ODD(1.00)[1];
ARC_ALLOW(-1.00)[google.com:s=arc-20240605:i=2];
DMARC_POLICY_ALLOW(-0.50)[gmail.com,none];
MANY_INVISIBLE_PARTS(0.20)[3];
R_DKIM_ALLOW(-0.20)[google.com:s=20230601,gmail.com:s=20230601];
R_SPF_ALLOW(-0.20)[+ip4:209.85.128.0/17:c];
MIME_GOOD(-0.10)[multipart/alternative,text/plain];
MIME_BASE64_TEXT(0.10)[];
FREEMAIL_ENVFROM(0.00)[gmail.com];
RCVD_TLS_LAST(0.00)[];
RCVD_COUNT_THREE(0.00)[3];
FREEMAIL_TO(0.00)[gmail.com];
TAGGED_FROM(0.00)[caf_=gmail=xinutv];
FREEMAIL_FROM(0.00)[gmail.com];
FREEMAIL_REPLYTO(0.00)[gmail.com];
MIME_TRACE(0.00)[0:+,1:+,2:~];
RCPT_COUNT_ONE(0.00)[1];
FORGED_SENDER(0.00)[tconvertino@gmail.com,couchmoney@gmail.com];
RCVD_IN_DNSWL_NONE(0.00)[209.85.220.73:received];
MISSING_XM_UA(0.00)[];
HAS_REPLYTO(0.00)[tconvertino@gmail.com];
TO_DN_NONE(0.00)[];
DNSWL_BLOCKED(0.00)[209.85.167.47:from];
FORGED_SENDER_FORWARDING(0.00)[];
FROM_NEQ_ENVFROM(0.00)[tconvertino@gmail.com,couchmoney@gmail.com];
DWL_DNSWL_NONE(0.00)[google.com:dkim];
DKIM_TRACE(0.00)[google.com:+,gmail.com:+];
DWL_DNSWL_BLOCKED(0.00)[gmail.com:dkim];
TO_DOM_EQ_FROM_DOM(0.00)[];
FROM_NO_DN(0.00)[];
FWD_GOOGLE(0.00)[couchmoney@gmail.com];
ASN(0.00)[asn:15169, ipnet:209.85.128.0/17, country:US];
RWL_MAILSPIKE_POSSIBLE(0.00)[209.85.167.47:from];
REPLYTO_EQ_FROM(0.00)[]
X-Rspamd-Action: no action
X-TUID: SJoEMVYLjYPK
--0000000000009e8d3e064896eeed
Content-Type: text/plain; charset="UTF-8"; format=flowed; delsp=yes
Content-Transfer-Encoding: base64
VGhpcyBldmVudCBoYXMgYmVlbiB1cGRhdGVkDQpDaGFuZ2VkOiB0aW1lDQoNCg0KcGFpbnRpbmcg
Y2xhc3MNClRodXJzZGF5IEZlYiAxMiwgMjAyNiDii4UgN3BtIOKAkyA5cG0NClBhY2lmaWMgVGlt
ZSAtIExvcyBBbmdlbGVzDQoNCg0KDQpPcmdhbml6ZXINCnRjb252ZXJ0aW5vQGdtYWlsLmNvbQ0K
dGNvbnZlcnRpbm9AZ21haWwuY29tDQoNCn5+Ly9+fg0KSW52aXRhdGlvbiBmcm9tIEdvb2dsZSBD
YWxlbmRhcjogaHR0cHM6Ly9jYWxlbmRhci5nb29nbGUuY29tL2NhbGVuZGFyLw0KDQpZb3UgYXJl
IHJlY2VpdmluZyB0aGlzIGVtYWlsIGJlY2F1c2UgeW91IGFyZSBzdWJzY3JpYmVkIHRvIGNhbGVu
ZGFyICANCm5vdGlmaWNhdGlvbnMuIFRvIHN0b3AgcmVjZWl2aW5nIHRoZXNlIGVtYWlscywgZ28g
dG8gIA0KaHR0cHM6Ly9jYWxlbmRhci5nb29nbGUuY29tL2NhbGVuZGFyL3Ivc2V0dGluZ3MsIHNl
bGVjdCB0aGlzIGNhbGVuZGFyLCBhbmQgIA0KY2hhbmdlICJPdGhlciBub3RpZmljYXRpb25zIi4N
Cg0KRm9yd2FyZGluZyB0aGlzIGludml0YXRpb24gY291bGQgYWxsb3cgYW55IHJlY2lwaWVudCB0
byBzZW5kIGEgcmVzcG9uc2UgdG8gIA0KdGhlIG9yZ2FuaXplciwgYmUgYWRkZWQgdG8gdGhlIGd1
ZXN0IGxpc3QsIGludml0ZSBvdGhlcnMgcmVnYXJkbGVzcyBvZiAgDQp0aGVpciBvd24gaW52aXRh
dGlvbiBzdGF0dXMsIG9yIG1vZGlmeSB5b3VyIFJTVlAuDQoNCkxlYXJuIG1vcmUgaHR0cHM6Ly9z
dXBwb3J0Lmdvb2dsZS5jb20vY2FsZW5kYXIvYW5zd2VyLzM3MTM1I2ZvcndhcmRpbmcNCg==
--0000000000009e8d3e064896eeed
Content-Type: text/html; charset="UTF-8"
Content-Transfer-Encoding: quoted-printable
<!doctype html><html xmlns=3D"http://www.w3.org/1999/xhtml" xmlns:v=3D"urn:=
schemas-microsoft-com:vml" xmlns:o=3D"urn:schemas-microsoft-com:office:offi=
ce"><head><title></title><!--[if !mso]><meta http-equiv=3D"X-UA-Compatible"=
content=3D"IE=3Dedge"><![endif]--><meta http-equiv=3D"Content-Type" conten=
t=3D"text/html; charset=3DUTF-8"><meta name=3D"viewport" content=3D"width=
=3Ddevice-width,initial-scale=3D1"><meta name=3D"color-scheme" content=3D"l=
ight dark"><meta name=3D"supported-color-schemes" content=3D"light dark">
<style>
body, html {
font-family: Roboto, Helvetica, Arial, sans-serif;
}
body {
margin: 0;
padding: 0;
-webkit-font-smoothing: antialiased;
-webkit-text-size-adjust: 100%;
-ms-text-size-adjust: 100%;
}
#outlook a {
padding: 0;
}
.ReadMsgBody {
width: 100%;
}
.ExternalClass {
width: 100%;
}
.ExternalClass * {
line-height: 100%;
}
table,
td {
mso-table-lspace: 0pt;
mso-table-rspace: 0pt;
}
img {
border: 0;
height: auto;
line-height: 100%;
outline: none;
text-decoration: none;
-ms-interpolation-mode: bicubic;
}
p {
display: block;
margin: 13px 0;
}
</style>
<!--[if !mso]><!-->
<style>
@media only screen and (max-width:580px) {
@-ms-viewport {
width: 320px;
}
@viewport {
width: 320px;
}
}
</style>
<!--<![endif]-->
<!--[if mso]>
<xml>
<o:OfficeDocumentSettings>
<o:AllowPNG/>
<o:PixelsPerInch>96</o:PixelsPerInch>
</o:OfficeDocumentSettings>
</xml>
<![endif]-->
<!--[if lte mso 11]>
<style>
.outlook-group-fix { width:100% !important; }
</style>
<![endif]-->
<!--[if !mso]><!-- -->
<style>body, html {font-family:Roboto,Helvetica,Arial,sans-serif;}@font-f=
ace {
font-family: 'Roboto';
font-style: normal;
font-weight: 400;
font-stretch: normal;
src: url(//fonts.gstatic.com/s/roboto/v48/KFOMCnqEu92Fr1ME7kSn66aGLdTylUA=
MQXC89YmC2DPNWubEbVmUiA8.ttf) format('truetype');
}
@font-face {
font-family: 'Roboto';
font-style: normal;
font-weight: 500;
font-stretch: normal;
src: url(//fonts.gstatic.com/s/roboto/v48/KFOMCnqEu92Fr1ME7kSn66aGLdTylUA=
MQXC89YmC2DPNWub2bVmUiA8.ttf) format('truetype');
}
@font-face {
font-family: 'Roboto';
font-style: normal;
font-weight: 700;
font-stretch: normal;
src: url(//fonts.gstatic.com/s/roboto/v48/KFOMCnqEu92Fr1ME7kSn66aGLdTylUA=
MQXC89YmC2DPNWuYjalmUiA8.ttf) format('truetype');
}
@font-face {
font-family: 'Material Icons Extended';
font-style: normal;
font-weight: 400;
src: url(//fonts.gstatic.com/s/materialiconsextended/v154/kJEjBvgX7BgnkSr=
UwT8UnLVc38YydejYY-oE_LvM.ttf) format('truetype');
}
@font-face {
font-family: 'Google Material Icons';
font-style: normal;
font-weight: 400;
src: url(//fonts.gstatic.com/s/googlematerialicons/v144/Gw6kwdfw6UnXLJCcm=
afZyFRXb3BL9rvi0QZG3g.otf) format('opentype');
}
.google-material-icons {
font-family: 'Google Material Icons';
font-weight: normal;
font-style: normal;
font-size: 24px;
line-height: 1;
letter-spacing: normal;
text-transform: none;
display: inline-block;
white-space: nowrap;
word-wrap: normal;
direction: ltr;
}
@font-face {
font-family: 'Google Material Icons Filled';
font-style: normal;
font-weight: 400;
src: url(//fonts.gstatic.com/s/googlematerialiconsfilled/v118/WWXFlimHYg6=
HKI3TavMkbKdhBmDvgach8TVpeGsuueSZJH4.otf) format('opentype');
}
.google-material-icons-filled {
font-family: 'Google Material Icons Filled';
font-weight: normal;
font-style: normal;
font-size: 24px;
line-height: 1;
letter-spacing: normal;
text-transform: none;
display: inline-block;
white-space: nowrap;
word-wrap: normal;
direction: ltr;
}
@font-face {
font-family: 'Google Sans';
font-style: normal;
font-weight: 400;
src: url(//fonts.gstatic.com/s/googlesans/v14/4UaGrENHsxJlGDuGo1OIlL3Owps=
.ttf) format('truetype');
}
@font-face {
font-family: 'Google Sans';
font-style: normal;
font-weight: 500;
src: url(//fonts.gstatic.com/s/googlesans/v14/4UabrENHsxJlGDuGo1OIlLU94Yt=
zCwM.ttf) format('truetype');
}
@font-face {
font-family: 'Google Sans';
font-style: normal;
font-weight: 700;
src: url(//fonts.gstatic.com/s/googlesans/v14/4UabrENHsxJlGDuGo1OIlLV154t=
zCwM.ttf) format('truetype');
}
</style><!--<![endif]-->
<style>
.body-container {
padding-left: 16px;
padding-right: 16px;
}
</style>
=20
<style>
u+.body .body-container,
body[data-outlook-cycle] .body-container,
#MessageViewBody .body-container {
padding-left: 0;
padding-right: 0;
}
</style>
=20
<style>
@media only screen and (min-width:580px) {
.column-per-37 {
width: 37% !important;
max-width: 37%;
}
.column-per-63 {
width: 63% !important;
max-width: 63%;
}
}
</style>
=20
<style>
.appointment-buttons th {
display: block;
clear: both;
float: left;
margin-top: 12px;
}
.appointment-buttons th a {
float: left;
}
#MessageViewBody .appointment-buttons th {
margin-top: 24px;
}
</style>
=20
<style>
@media only screen and (max-width:580px) {
table.full-width-mobile {
width: 100% !important;
}
td.full-width-mobile {
width: auto !important;
}
}
</style>
<style>
.main-container-inner,
.info-bar-inner {
padding: 12px 16px !important;
}
.main-column-table-ltr {
padding-right: 0 !important;
}
.main-column-table-rtl {
padding-left: 0 !important;
}
@media only screen and (min-width:580px) {
.main-container-inner {
padding: 24px 32px !important;
}
.info-bar-inner {
padding: 12px 32px !important;
}
.main-column-table-ltr {
padding-right: 32px !important;
}
.main-column-table-rtl {
padding-left: 32px !important;
}
.appointment-buttons th {
display: table-cell;
clear: none;
}
}
.primary-text {
color: #3c4043 !important;
}
.secondary-text,
.phone-number a {
color: #70757a !important;
}
.accent-text {
color: #1a73e8 !important;
}
.accent-text-dark {
color: #185abc !important;
}
.grey-button-text,
.attachment-chip a {
color: #5f6368 !important;
}
.primary-button {
background-color: #1a73e8 !important;
}
.primary-button-text {
color: #fff !important;
}
.underline-on-hover:hover {
text-decoration: underline !important;
}
.grey-infobar-text {
color: #202124 !important;
}
@media (prefers-color-scheme: dark) {
.primary-text:not([class^=3D"x_"]) {
color: #e8eaed !important;
}
.secondary-text:not([class^=3D"x_"]),
.phone-number:not([class^=3D"x_"]) a {
color: #9aa0a6 !important;
}
.grey-button-text:not([class^=3D"x_"]),
.attachment-chip:not([class^=3D"x_"]) a {
color: #bdc1c6 !important;
}
.accent-text:not([class^=3D"x_"]),
.hairline-button-text:not([class^=3D"x_"]) {
color: #8ab4f8 !important;
}
.primary-button:not([class^=3D"x_"]) {
background-color: #8ab4f8 !important;
}
.primary-button-text:not([class^=3D"x_"]) {
color: #202124 !important;
}
}
</style>
<style>
@media (prefers-color-scheme: dark) {
.cse-banner:not([class^=3D"x_"]) {
background-color: #3c4043 !important; /* Google Grey 800 */
}
.encryption-icon:not([class^=3D"x_"]) {
/* WARNING: This causes the whole style tag to get stripped in Gm=
ail. */
background-image: url('https://fonts.gstatic.com/s/i/googlemateri=
aliconsfilled/encrypted/v3/gm_grey200-24dp/2x/gm_filled_encrypted_gm_grey20=
0_24dp.png') !important;
}
}
</style>
<!--[if !mso]><!-->
<style>
.prevent-link a {
color: inherit !important;
text-decoration: none !important;
font-size: inherit !important;
font-family: inherit !important;
font-weight: inherit !important;
line-height: inherit !important;
}
</style>
<!--<![endif]-->
<!--[if mso | IE]>
<style>
.main-container-inner {
padding: 24px 32px !important;
}
.info-bar-inner {
padding: 12px 32px !important;
}
.cse-banner .encryption-icon {
/* We use the IE workaround instead. */
background-image: none !important;
}
.cse-banner .encryption-icon .ms-fallback {
display: block !important;
}
/* NB: Some MS clients ignore dark-scheme styling and apply their o=
wn, so there's nothing we can do to help there. */
@media (prefers-color-scheme: dark) {
.cse-banner:not([class^=3D"x_"]) .encryption-icon .ms-fallback {
display: none !important;
}
.cse-banner:not([class^=3D"x_"]) .encryption-icon .ms-fallback-da=
rk {
display: block !important;
}
}
</style>
<![endif]-->
</head><body class=3D"body"><span itemscope itemtype=3D"http://schema.org=
/InformAction"><span itemprop=3D"object" itemscope itemtype=3D"http://schem=
a.org/Event"><meta itemprop=3D"eventStatus" content=3D"http://schema.org/Ev=
entRescheduled"/><span itemprop=3D"publisher" itemscope itemtype=3D"http://=
schema.org/Organization"><meta itemprop=3D"name" content=3D"Google Calendar=
"/></span><meta itemprop=3D"eventId/googleCalendar" content=3D"4octcgvhijqj=
m45h8d9dr4iq1m"/><span style=3D"display: none; font-size: 1px; color: #fff;=
line-height: 1px; height: 0; max-height: 0; width: 0; max-width: 0; opacit=
y: 0; overflow: hidden;" itemprop=3D"name">painting class</span><meta itemp=
rop=3D"url" content=3D"https://calendar.google.com/calendar/r?eid=3DNG9jdGN=
ndmhpanFqbTQ1aDhkOWRyNGlxMW0gdGNvbnZlcnRpbm9AbQ&amp;es=3D1"/><span aria-hid=
den=3D"true"><time itemprop=3D"startDate" datetime=3D"20260213T030000Z"></t=
ime><time itemprop=3D"endDate" datetime=3D"20260213T050000Z"></time></span>=
<div style=3D"display: none; font-size: 1px; color: #fff; line-height: 1px;=
height: 0; max-height: 0; width: 0; max-width: 0; opacity: 0; overflow: hi=
dden;">You have been invited by tconvertino@gmail.com to attend an event na=
med painting class on Thursday Feb 12, 2026 =E2=8B=85 7pm =E2=80=93 9pm (Pa=
cific Time - Los Angeles).</div><table border=3D"0" cellpadding=3D"0" cells=
pacing=3D"0" role=3D"presentation" align=3D"center" style=3D"width:100%;" c=
lass=3D"body-container"><tbody><tr><td style=3D"" class=3D"" align=3D"left"=
><!--[if mso | IE]><table border=3D"0" cellpadding=3D"0" cellspacing=3D"0" =
role=3D"presentation"><tr><td height=3D"16" style=3D"height:16px;"><![endif=
]--><div style=3D"height:16px;" aria-hidden=3D"true"> &nbsp; </div><!--[if =
mso | IE]></td></tr></table><![endif]--><table border=3D"0" cellpadding=3D"=
0" cellspacing=3D"0" role=3D"presentation" align=3D"center" style=3D"width:=
100%;" class=3D""><tbody><tr><td style=3D"background-color: #e6f4ea;color: =
#0d5327;padding: 12px 32px; border-radius: 8px;font-family: Roboto, sans-se=
rif;font-size: 14px; line-height: 20px;text-align: left;" class=3D"info-bar=
-inner"><span style=3D"font-weight: 700;">This event has been updated</span=
><br/><span style=3D"display:none" itemprop=3D"about" itemscope itemtype=3D=
"http://schema.org/Thing/Clock"><meta itemprop=3D"description" content=3D"T=
ime updated"/></span><div style=3D""><span style=3D"font-weight: 700;">Chan=
ged:</span> time</div></td></tr></tbody></table><!--[if mso | IE]><table bo=
rder=3D"0" cellpadding=3D"0" cellspacing=3D"0" role=3D"presentation"><tr><t=
d height=3D"12" style=3D"height:12px;"><![endif]--><div style=3D"height:12p=
x;" aria-hidden=3D"true"> &nbsp; </div><!--[if mso | IE]></td></tr></table>=
<![endif]--><table border=3D"0" cellpadding=3D"0" cellspacing=3D"0" role=3D=
"presentation" align=3D"center" style=3D"width:100%;" class=3D""><tbody><tr=
><td style=3D"border: solid 1px #dadce0; border-radius: 8px; direction: rtl=
; font-size: 0; padding: 24px 32px; text-align: left; vertical-align: top;"=
class=3D"main-container-inner"><!--[if mso | IE]><table border=3D"0" cellp=
adding=3D"0" cellspacing=3D"0" role=3D"presentation"><tr><![endif]--><div c=
lass=3D"" style=3D"font-size: 13px; text-align: left; direction: ltr; displ=
ay: inline-block; vertical-align: top; width: 100%;overflow: hidden; word-w=
rap: break-word;"><table border=3D"0" cellpadding=3D"0" cellspacing=3D"0" r=
ole=3D"presentation" width=3D"100%" class=3D"main-column-table-ltr" style=
=3D"padding-right: 32px; padding-left: 0;;table-layout: fixed;"><tbody><tr>=
<td class=3D"main-column-td" style=3D"padding:0; vertical-align:top;"><tabl=
e border=3D"0" cellpadding=3D"0" cellspacing=3D"0" role=3D"presentation" wi=
dth=3D"100%" style=3D"table-layout: fixed;"><tr><td style=3D"font-size: 0; =
padding: 0; text-align: left; word-break: break-word;;padding-bottom:24px;"=
><div style=3D"font-family: Roboto, sans-serif;font-style: normal; font-wei=
ght: 400; font-size: 14px; line-height: 20px; letter-spacing: 0.2px;color: =
#3c4043; text-decoration: none;" class=3D"primary-text" role=3D"presentatio=
n"><span aria-hidden=3D"true"><time itemprop=3D"startDate" datetime=3D"2026=
0213T030000Z"></time><time itemprop=3D"endDate" datetime=3D"20260213T050000=
Z"></time></span><table border=3D"0" cellpadding=3D"0" cellspacing=3D"0" ro=
le=3D"presentation" style=3D"padding-bottom: 4px;"><tr><td><h2 class=3D"pri=
mary-text" style=3D"font-size: 14px;color: #3c4043; text-decoration: none;f=
ont-weight: 700;-webkit-font-smoothing: antialiased;margin: 0; padding: 0;"=
>When</h2></td><td style=3D"width: 8px;"></td><td style=3D"padding-top: 2px=
; padding-bottom: 3px;"><div style=3D"background-color: #1e8e3e; border-rad=
ius: 10px; padding: 1px 5px; line-height: 13px;"><span style=3D"color: whit=
e; font-size: 11px; font-weight: 700; letter-spacing: 0.8px; text-transform=
: uppercase; vertical-align: top;">CHANGED</span></div></td></tr></table><s=
pan>Thursday Feb 12, 2026 =E2=8B=85 7pm =E2=80=93 9pm (Pacific Time - Los A=
ngeles)<br/><span style=3D"text-decoration: line-through;"><del><span style=
=3D"display: none; font-size: 1px; color: #fff; line-height: 1px; height: 0=
; max-height: 0; width: 0; max-width: 0; opacity: 0; overflow: hidden;font-=
size: 0; display: block;">Old: </span>Thursday Jan 22, 2026 =E2=8B=85 7pm =
=E2=80=93 9pm (Pacific Time - Los Angeles)</del></span></span></div></td></=
tr><tr><td style=3D"font-size: 0; padding: 0; text-align: left; word-break:=
break-word;;padding-bottom:24px;"><div style=3D"font-family: Roboto, sans-=
serif;font-style: normal; font-weight: 400; font-size: 14px; line-height: 2=
0px; letter-spacing: 0.2px;color: #3c4043; text-decoration: none;" class=3D=
"primary-text" role=3D"presentation"><table border=3D"0" cellpadding=3D"0" =
cellspacing=3D"0" role=3D"presentation" style=3D"padding-bottom: 4px;"><tr>=
<td><h2 class=3D"primary-text" style=3D"font-size: 14px;color: #3c4043; tex=
t-decoration: none;font-weight: 700;-webkit-font-smoothing: antialiased;mar=
gin: 0; padding: 0;">Organizer</h2></td></tr></table><div style=3D"color: #=
3c4042;"><span class=3D"notranslate"><a class=3D"primary-text underline-on-=
hover" style=3D"display: inline-block;;color: #3c4043; text-decoration: non=
e;" href=3D"mailto:tconvertino@gmail.com">tconvertino@gmail.com</a></span><=
span itemprop=3D"organizer" itemscope itemtype=3D"http://schema.org/Person"=
><meta itemprop=3D"name" content=3D"tconvertino@gmail.com"/><meta itemprop=
=3D"email" content=3D"tconvertino@gmail.com"/></span></div></div></td></tr>=
<tr><td style=3D"font-size: 0; padding: 0; text-align: left; word-break: br=
eak-word;;padding-bottom:24px;"><div style=3D"font-family: Roboto, sans-ser=
if;font-style: normal; font-weight: 400; font-size: 14px; line-height: 20px=
; letter-spacing: 0.2px;color: #3c4043; text-decoration: none;" class=3D"pr=
imary-text" role=3D"presentation"><table border=3D"0" cellpadding=3D"0" cel=
lspacing=3D"0" role=3D"presentation" style=3D"padding-bottom: 4px;"><tr><td=
><h2 class=3D"primary-text" style=3D"font-size: 14px;color: #3c4043; text-d=
ecoration: none;font-weight: 700;-webkit-font-smoothing: antialiased;margin=
: 0; padding: 0;">Guests</h2></td></tr></table><div style=3D"padding-bottom=
: 4px; text-align: left;;color: #3c4042;"></div><a href=3D"https://calendar=
.google.com/calendar/r?eid=3DNG9jdGNndmhpanFqbTQ1aDhkOWRyNGlxMW0gdGNvbnZlcn=
Rpbm9AbQ&amp;es=3D1" style=3D"display: inline-block;;color: #1a73e8; text-d=
ecoration: none;font-weight: 700;" target=3D"_blank" class=3D"accent-text u=
nderline-on-hover">View all guest info</a></div></td></tr></table></td></tr=
></tbody></table></div><!--[if mso | IE]></tr></table><![endif]--></td></tr=
></tbody></table><table border=3D"0" cellpadding=3D"0" cellspacing=3D"0" ro=
le=3D"presentation" align=3D"center" style=3D"width:100%;" class=3D""><tbod=
y><tr><td style=3D"font-size: 0; padding: 0; text-align: left; word-break: =
break-word;;padding:4px 12px;" class=3D"" align=3D"left"><div class=3D"seco=
ndary-text" style=3D"color: #70757a; text-decoration: none;font-family: Rob=
oto, sans-serif;font-size: 12px; line-height: 16px; mso-line-height-rule: e=
xactly; text-align: left;"><p>Invitation from <a href=3D"https://calendar.g=
oogle.com/calendar/" class=3D"accent-text underline-on-hover" style=3D"font=
-family: Roboto, sans-serif;font-size: 12px; line-height: 16px; mso-line-he=
ight-rule: exactly;;color: #1a73e8; text-decoration: none;" target=3D"_blan=
k">Google Calendar</a></p><p>You are receiving this email because you are s=
ubscribed to calendar notifications. To stop receiving these emails, go to =
<a href=3D"https://calendar.google.com/calendar/r/settings" class=3D"accent=
-text underline-on-hover" style=3D"font-family: Roboto, sans-serif;font-siz=
e: 12px; line-height: 16px; mso-line-height-rule: exactly;;color: #1a73e8; =
text-decoration: none;" target=3D"_blank">Calendar settings</a>, select thi=
s calendar, and change "Other notifications".</p><p>Forwarding this invitat=
ion could allow any recipient to send a response to the organizer, be added=
to the guest list, invite others regardless of their own invitation status=
, or modify your RSVP. <a class=3D"accent-text underline-on-hover" style=3D=
"font-family: Roboto, sans-serif;font-size: 12px; line-height: 16px; mso-li=
ne-height-rule: exactly;;color: #1a73e8; text-decoration: none;" href=3D"ht=
tps://support.google.com/calendar/answer/37135#forwarding">Learn more</a></=
p></div></td></tr></tbody></table></td></tr></tbody></table></span></span><=
/body></html>
--0000000000009e8d3e064896eeed--

View File

@@ -24,7 +24,7 @@ serde = { version = "1.0.219", features = ["derive"] }
itertools = "0.14.0"
serde_json = { version = "1.0.140", features = ["unbounded_depth"] }
chrono = "0.4.40"
graphql_client = "0.14.0"
graphql_client = "0.16.0"
thiserror = "2.0.12"
gloo-net = { version = "0.6.0", features = ["json", "serde_json"] }
human_format = "1.1.0"

View File

@@ -123,10 +123,10 @@ pub fn view(model: &Model) -> Node<Msg> {
} => {
if let Some(catchup) = &model.catchup {
catchup_view(
news_post(post, &model.content_el, true, model.read_completion_ratio),
news_post(post, &model.content_el, true, 0.),
&catchup.items,
is_loading,
0.,
model.read_completion_ratio,
)
} else {
normal_view(
@@ -351,8 +351,8 @@ fn search_results(
attrs! {
At::Href => urls::thread(&tid)
},
div![title_break, &r.subject],
span![C!["text-xs"], pretty_authors(&r.authors)],
div![C!["line-clamp-2"], title_break, &r.subject],
span![C!["line-clamp-2", "text-xs"], pretty_authors(&r.authors)],
div![
C!["flex", "flex-wrap", "justify-between"],
span![tags_chiclet(&tags)],