Hex Artifact Content
Logged in as anonymous

Artifact 1e3d90a7ddf724d5b4f62147f9f75b5ca6183f3189da7fd35c0a572e1ee9514a:


0000: 75 73 65 20 63 72 61 74 65 3a 3a 7b 0a 09 41 72  use crate::{..Ar
0010: 63 2c 0a 09 63 6f 6d 6d 61 6e 64 2c 0a 09 4d 75  c,..command,..Mu
0020: 74 65 78 2c 0a 09 73 71 6c 3a 3a 44 62 2c 0a 09  tex,..sql::Db,..
0030: 74 67 5f 62 6f 74 3a 3a 7b 0a 09 09 4d 79 4d 65  tg_bot::{...MyMe
0040: 73 73 61 67 65 2c 0a 09 09 54 67 2c 0a 09 7d 2c  ssage,...Tg,..},
0050: 0a 7d 3b 0a 0a 75 73 65 20 73 74 64 3a 3a 7b 0a  .};..use std::{.
0060: 09 62 6f 72 72 6f 77 3a 3a 43 6f 77 2c 0a 09 63  .borrow::Cow,..c
0070: 6f 6c 6c 65 63 74 69 6f 6e 73 3a 3a 7b 0a 09 09  ollections::{...
0080: 42 54 72 65 65 4d 61 70 2c 0a 09 09 48 61 73 68  BTreeMap,...Hash
0090: 4d 61 70 2c 0a 09 09 48 61 73 68 53 65 74 2c 0a  Map,...HashSet,.
00a0: 09 7d 2c 0a 09 74 69 6d 65 3a 3a 44 75 72 61 74  .},..time::Durat
00b0: 69 6f 6e 2c 0a 7d 3b 0a 0a 75 73 65 20 61 73 79  ion,.};..use asy
00c0: 6e 63 5f 63 6f 6d 70 61 74 3a 3a 43 6f 6d 70 61  nc_compat::Compa
00d0: 74 3b 0a 75 73 65 20 63 68 72 6f 6e 6f 3a 3a 7b  t;.use chrono::{
00e0: 0a 09 44 61 74 65 54 69 6d 65 2c 0a 09 4c 6f 63  ..DateTime,..Loc
00f0: 61 6c 2c 0a 7d 3b 0a 75 73 65 20 6c 61 7a 79 5f  al,.};.use lazy_
0100: 73 74 61 74 69 63 3a 3a 6c 61 7a 79 5f 73 74 61  static::lazy_sta
0110: 74 69 63 3b 0a 75 73 65 20 72 65 67 65 78 3a 3a  tic;.use regex::
0120: 52 65 67 65 78 3b 0a 75 73 65 20 72 65 71 77 65  Regex;.use reqwe
0130: 73 74 3a 3a 68 65 61 64 65 72 3a 3a 4c 41 53 54  st::header::LAST
0140: 5f 4d 4f 44 49 46 49 45 44 3b 0a 75 73 65 20 73  _MODIFIED;.use s
0150: 6d 6f 6c 3a 3a 54 69 6d 65 72 3b 0a 75 73 65 20  mol::Timer;.use 
0160: 73 74 61 63 6b 65 64 5f 65 72 72 6f 72 73 3a 3a  stacked_errors::
0170: 7b 0a 09 52 65 73 75 6c 74 2c 0a 09 53 74 61 63  {..Result,..Stac
0180: 6b 61 62 6c 65 45 72 72 2c 0a 09 61 6e 79 68 6f  kableErr,..anyho
0190: 77 2c 0a 09 62 61 69 6c 2c 0a 7d 3b 0a 75 73 65  w,..bail,.};.use
01a0: 20 74 67 62 6f 74 3a 3a 7b 0a 09 68 61 6e 64 6c   tgbot::{..handl
01b0: 65 72 3a 3a 55 70 64 61 74 65 48 61 6e 64 6c 65  er::UpdateHandle
01c0: 72 2c 0a 09 74 79 70 65 73 3a 3a 7b 0a 09 09 43  r,..types::{...C
01d0: 68 61 74 50 65 65 72 49 64 2c 0a 09 09 43 6f 6d  hatPeerId,...Com
01e0: 6d 61 6e 64 2c 0a 09 09 55 70 64 61 74 65 2c 0a  mand,...Update,.
01f0: 09 09 55 70 64 61 74 65 54 79 70 65 2c 0a 09 09  ..UpdateType,...
0200: 55 73 65 72 50 65 65 72 49 64 2c 0a 09 7d 2c 0a  UserPeerId,..},.
0210: 7d 3b 0a 75 73 65 20 74 74 6c 5f 63 61 63 68 65  };.use ttl_cache
0220: 3a 3a 54 74 6c 43 61 63 68 65 3b 0a 0a 6c 61 7a  ::TtlCache;..laz
0230: 79 5f 73 74 61 74 69 63 21 7b 0a 09 70 75 62 20  y_static!{..pub 
0240: 73 74 61 74 69 63 20 72 65 66 20 52 45 5f 53 50  static ref RE_SP
0250: 45 43 49 41 4c 3a 20 52 65 67 65 78 20 3d 20 52  ECIAL: Regex = R
0260: 65 67 65 78 3a 3a 6e 65 77 28 72 22 28 5b 5c 2d  egex::new(r"([\-
0270: 5f 2a 5c 5b 5c 5d 28 29 7e 60 3e 23 2b 7c 7b 7d  _*\[\]()~`>#+|{}
0280: 5c 2e 21 5d 29 22 29 2e 75 6e 77 72 61 70 28 29  \.!])").unwrap()
0290: 3b 0a 7d 0a 0a 2f 2f 2f 20 45 73 63 61 70 65 20  ;.}../// Escape 
02a0: 63 68 61 72 61 63 74 65 72 73 20 74 68 61 74 20  characters that 
02b0: 61 72 65 20 73 70 65 63 69 61 6c 20 69 6e 20 54  are special in T
02c0: 65 6c 65 67 72 61 6d 20 4d 61 72 6b 64 6f 77 6e  elegram Markdown
02d0: 56 32 20 62 79 20 70 72 65 66 69 78 69 6e 67 20  V2 by prefixing 
02e0: 74 68 65 6d 20 77 69 74 68 20 61 20 62 61 63 6b  them with a back
02f0: 73 6c 61 73 68 2e 0a 2f 2f 2f 0a 2f 2f 2f 20 54  slash..///./// T
0300: 68 69 73 20 65 6e 73 75 72 65 73 20 74 68 65 20  his ensures the 
0310: 72 65 74 75 72 6e 65 64 20 73 74 72 69 6e 67 20  returned string 
0320: 63 61 6e 20 62 65 20 75 73 65 64 20 61 73 20 4d  can be used as M
0330: 61 72 6b 64 6f 77 6e 56 32 2d 66 6f 72 6d 61 74  arkdownV2-format
0340: 74 65 64 20 54 65 6c 65 67 72 61 6d 20 6d 65 73  ted Telegram mes
0350: 73 61 67 65 20 63 6f 6e 74 65 6e 74 0a 2f 2f 2f  sage content.///
0360: 20 77 69 74 68 6f 75 74 20 73 70 65 63 69 61 6c   without special
0370: 20 63 68 61 72 61 63 74 65 72 73 20 62 65 69 6e   characters bein
0380: 67 20 69 6e 74 65 72 70 72 65 74 65 64 20 61 73  g interpreted as
0390: 20 4d 61 72 6b 64 6f 77 6e 56 32 20 6d 61 72 6b   MarkdownV2 mark
03a0: 75 70 2e 0a 70 75 62 20 66 6e 20 65 6e 63 6f 64  up..pub fn encod
03b0: 65 20 28 74 65 78 74 3a 20 26 73 74 72 29 20 2d  e (text: &str) -
03c0: 3e 20 43 6f 77 3c 27 5f 2c 20 73 74 72 3e 20 7b  > Cow<'_, str> {
03d0: 0a 09 52 45 5f 53 50 45 43 49 41 4c 2e 72 65 70  ..RE_SPECIAL.rep
03e0: 6c 61 63 65 5f 61 6c 6c 28 74 65 78 74 2c 20 22  lace_all(text, "
03f0: 5c 5c 24 31 22 29 0a 7d 0a 0a 2f 2f 20 54 68 69  \\$1").}..// Thi
0400: 73 20 6f 6e 65 20 64 6f 65 73 20 6e 6f 74 68 69  s one does nothi
0410: 6e 67 20 65 78 63 65 70 74 20 6d 61 6b 69 6e 67  ng except making
0420: 20 73 75 72 65 20 6f 6e 6c 79 20 6f 6e 65 20 74   sure only one t
0430: 6f 6b 65 6e 20 65 78 69 73 74 73 20 66 6f 72 20  oken exists for 
0440: 65 61 63 68 20 69 64 0a 70 75 62 20 73 74 72 75  each id.pub stru
0450: 63 74 20 54 6f 6b 65 6e 20 7b 0a 09 72 75 6e 6e  ct Token {..runn
0460: 69 6e 67 3a 20 41 72 63 3c 4d 75 74 65 78 3c 48  ing: Arc<Mutex<H
0470: 61 73 68 53 65 74 3c 69 33 32 3e 3e 3e 2c 0a 09  ashSet<i32>>>,..
0480: 6d 79 5f 69 64 3a 20 69 33 32 2c 0a 7d 0a 0a 69  my_id: i32,.}..i
0490: 6d 70 6c 20 54 6f 6b 65 6e 20 7b 0a 09 2f 2f 2f  mpl Token {..///
04a0: 20 41 74 74 65 6d 70 74 73 20 74 6f 20 61 63 71   Attempts to acq
04b0: 75 69 72 65 20 61 20 70 65 72 2d 69 64 20 74 6f  uire a per-id to
04c0: 6b 65 6e 20 62 79 20 69 6e 73 65 72 74 69 6e 67  ken by inserting
04d0: 20 60 6d 79 5f 69 64 60 20 69 6e 74 6f 20 74 68   `my_id` into th
04e0: 65 20 73 68 61 72 65 64 20 60 72 75 6e 6e 69 6e  e shared `runnin
04f0: 67 60 20 73 65 74 2e 0a 09 2f 2f 2f 0a 09 2f 2f  g` set...///..//
0500: 2f 20 49 66 20 74 68 65 20 69 64 20 77 61 73 20  / If the id was 
0510: 6e 6f 74 20 61 6c 72 65 61 64 79 20 70 72 65 73  not already pres
0520: 65 6e 74 2c 20 74 68 65 20 66 75 6e 63 74 69 6f  ent, the functio
0530: 6e 20 69 6e 73 65 72 74 73 20 69 74 20 61 6e 64  n inserts it and
0540: 20 72 65 74 75 72 6e 73 20 60 53 6f 6d 65 28 54   returns `Some(T
0550: 6f 6b 65 6e 29 60 2e 0a 09 2f 2f 2f 20 57 68 65  oken)`.../// Whe
0560: 6e 20 74 68 65 20 72 65 74 75 72 6e 65 64 20 60  n the returned `
0570: 54 6f 6b 65 6e 60 20 69 73 20 64 72 6f 70 70 65  Token` is droppe
0580: 64 2c 20 74 68 65 20 69 64 20 77 69 6c 6c 20 62  d, the id will b
0590: 65 20 72 65 6d 6f 76 65 64 20 66 72 6f 6d 20 74  e removed from t
05a0: 68 65 20 60 72 75 6e 6e 69 6e 67 60 20 73 65 74  he `running` set
05b0: 2c 0a 09 2f 2f 2f 20 61 6c 6c 6f 77 69 6e 67 20  ,../// allowing 
05c0: 73 75 62 73 65 71 75 65 6e 74 20 61 63 71 75 69  subsequent acqui
05d0: 73 69 74 69 6f 6e 73 20 66 6f 72 20 74 68 65 20  sitions for the 
05e0: 73 61 6d 65 20 69 64 2e 0a 09 2f 2f 2f 0a 09 2f  same id...///../
05f0: 2f 2f 20 23 20 50 61 72 61 6d 65 74 65 72 73 0a  // # Parameters.
0600: 09 2f 2f 2f 0a 09 2f 2f 2f 20 2d 20 60 72 75 6e  .///../// - `run
0610: 6e 69 6e 67 60 3a 20 53 68 61 72 65 64 20 73 65  ning`: Shared se
0620: 74 20 74 72 61 63 6b 69 6e 67 20 61 63 74 69 76  t tracking activ
0630: 65 20 69 64 73 2e 0a 09 2f 2f 2f 20 2d 20 60 6d  e ids.../// - `m
0640: 79 5f 69 64 60 3a 20 49 64 65 6e 74 69 66 69 65  y_id`: Identifie
0650: 72 20 74 6f 20 61 63 71 75 69 72 65 20 61 20 74  r to acquire a t
0660: 6f 6b 65 6e 20 66 6f 72 2e 0a 09 2f 2f 2f 0a 09  oken for...///..
0670: 2f 2f 2f 20 23 20 52 65 74 75 72 6e 73 0a 09 2f  /// # Returns../
0680: 2f 2f 0a 09 2f 2f 2f 20 60 4f 6b 28 54 6f 6b 65  //../// `Ok(Toke
0690: 6e 29 60 20 69 66 20 74 68 65 20 69 64 20 77 61  n)` if the id wa
06a0: 73 20 73 75 63 63 65 73 73 66 75 6c 6c 79 20 61  s successfully a
06b0: 63 71 75 69 72 65 64 2c 20 60 45 72 72 6f 72 60  cquired, `Error`
06c0: 20 69 66 20 61 20 74 6f 6b 65 6e 20 66 6f 72 20   if a token for 
06d0: 74 68 65 20 69 64 20 69 73 20 61 6c 72 65 61 64  the id is alread
06e0: 79 20 61 63 74 69 76 65 2e 0a 09 61 73 79 6e 63  y active...async
06f0: 20 66 6e 20 6e 65 77 20 28 72 75 6e 6e 69 6e 67   fn new (running
0700: 3a 20 26 41 72 63 3c 4d 75 74 65 78 3c 48 61 73  : &Arc<Mutex<Has
0710: 68 53 65 74 3c 69 33 32 3e 3e 3e 2c 20 6d 79 5f  hSet<i32>>>, my_
0720: 69 64 3a 20 69 33 32 29 20 2d 3e 20 52 65 73 75  id: i32) -> Resu
0730: 6c 74 3c 54 6f 6b 65 6e 3e 20 7b 0a 09 09 6c 65  lt<Token> {...le
0740: 74 20 72 75 6e 6e 69 6e 67 20 3d 20 72 75 6e 6e  t running = runn
0750: 69 6e 67 2e 63 6c 6f 6e 65 28 29 3b 0a 09 09 6c  ing.clone();...l
0760: 65 74 20 6d 75 74 20 73 65 74 20 3d 20 72 75 6e  et mut set = run
0770: 6e 69 6e 67 2e 6c 6f 63 6b 5f 61 72 63 28 29 2e  ning.lock_arc().
0780: 61 77 61 69 74 3b 0a 09 09 69 66 20 73 65 74 2e  await;...if set.
0790: 63 6f 6e 74 61 69 6e 73 28 26 6d 79 5f 69 64 29  contains(&my_id)
07a0: 20 7b 0a 09 09 09 62 61 69 6c 21 28 22 54 6f 6b   {....bail!("Tok
07b0: 65 6e 20 61 6c 72 65 61 64 79 20 74 61 6b 65 6e  en already taken
07c0: 22 29 3b 0a 09 09 7d 20 65 6c 73 65 20 7b 0a 09  ");...} else {..
07d0: 09 09 73 65 74 2e 69 6e 73 65 72 74 28 6d 79 5f  ..set.insert(my_
07e0: 69 64 29 3b 0a 09 09 09 4f 6b 28 54 6f 6b 65 6e  id);....Ok(Token
07f0: 20 7b 0a 09 09 09 09 72 75 6e 6e 69 6e 67 2c 0a   {.....running,.
0800: 09 09 09 09 6d 79 5f 69 64 2c 0a 09 09 09 7d 29  ....my_id,....})
0810: 0a 09 09 7d 0a 09 7d 0a 7d 0a 0a 69 6d 70 6c 20  ...}..}.}..impl 
0820: 44 72 6f 70 20 66 6f 72 20 54 6f 6b 65 6e 20 7b  Drop for Token {
0830: 0a 09 2f 2f 2f 20 52 65 6c 65 61 73 65 73 20 74  ../// Releases t
0840: 68 69 73 20 74 6f 6b 65 6e 27 73 20 63 6c 61 69  his token's clai
0850: 6d 20 6f 6e 20 74 68 65 20 73 68 61 72 65 64 20  m on the shared 
0860: 72 75 6e 6e 69 6e 67 2d 73 65 74 20 77 68 65 6e  running-set when
0870: 20 74 68 65 20 74 6f 6b 65 6e 20 69 73 20 64 72   the token is dr
0880: 6f 70 70 65 64 2e 0a 09 2f 2f 2f 0a 09 2f 2f 2f  opped...///..///
0890: 20 54 68 65 20 74 6f 6b 65 6e 27 73 20 69 64 65   The token's ide
08a0: 6e 74 69 66 69 65 72 20 69 73 20 72 65 6d 6f 76  ntifier is remov
08b0: 65 64 20 66 72 6f 6d 20 74 68 65 20 73 68 61 72  ed from the shar
08c0: 65 64 20 60 72 75 6e 6e 69 6e 67 60 20 73 65 74  ed `running` set
08d0: 20 73 6f 20 74 68 61 74 20 66 75 74 75 72 65 0a   so that future.
08e0: 09 2f 2f 2f 20 6f 70 65 72 61 74 69 6f 6e 73 20  ./// operations 
08f0: 66 6f 72 20 74 68 65 20 73 61 6d 65 20 69 64 20  for the same id 
0900: 6d 61 79 20 70 72 6f 63 65 65 64 2e 0a 09 2f 2f  may proceed...//
0910: 2f 0a 09 2f 2f 2f 20 54 4f 44 4f 3a 20 69 73 20  /../// TODO: is 
0920: 75 73 69 6e 67 20 62 6c 6f 63 6b 5f 6f 6e 20 69  using block_on i
0930: 6e 73 69 64 65 20 62 6c 6f 63 6b 5f 6f 6e 20 73  nside block_on s
0940: 61 66 65 3f 20 43 75 72 72 65 6e 74 6c 79 20 74  afe? Currently t
0950: 65 73 74 65 64 20 61 6e 64 20 77 6f 72 6b 69 6e  ested and workin
0960: 67 20 66 69 6e 65 2e 0a 09 66 6e 20 64 72 6f 70  g fine...fn drop
0970: 20 28 26 6d 75 74 20 73 65 6c 66 29 20 7b 0a 09   (&mut self) {..
0980: 09 73 6d 6f 6c 3a 3a 62 6c 6f 63 6b 5f 6f 6e 28  .smol::block_on(
0990: 61 73 79 6e 63 20 7b 0a 09 09 09 6c 65 74 20 6d  async {....let m
09a0: 75 74 20 73 65 74 20 3d 20 73 65 6c 66 2e 72 75  ut set = self.ru
09b0: 6e 6e 69 6e 67 2e 6c 6f 63 6b 5f 61 72 63 28 29  nning.lock_arc()
09c0: 2e 61 77 61 69 74 3b 0a 09 09 09 73 65 74 2e 72  .await;....set.r
09d0: 65 6d 6f 76 65 28 26 73 65 6c 66 2e 6d 79 5f 69  emove(&self.my_i
09e0: 64 29 3b 0a 09 09 7d 29 0a 09 7d 0a 7d 0a 0a 74  d);...})..}.}..t
09f0: 79 70 65 20 46 65 65 64 4c 69 73 74 20 3d 20 48  ype FeedList = H
0a00: 61 73 68 4d 61 70 3c 69 33 32 2c 20 53 74 72 69  ashMap<i32, Stri
0a10: 6e 67 3e 3b 0a 74 79 70 65 20 55 73 65 72 43 61  ng>;.type UserCa
0a20: 63 68 65 20 3d 20 54 74 6c 43 61 63 68 65 3c 69  che = TtlCache<i
0a30: 36 34 2c 20 41 72 63 3c 4d 75 74 65 78 3c 46 65  64, Arc<Mutex<Fe
0a40: 65 64 4c 69 73 74 3e 3e 3e 3b 0a 0a 23 5b 64 65  edList>>>;..#[de
0a50: 72 69 76 65 28 43 6c 6f 6e 65 29 5d 0a 70 75 62  rive(Clone)].pub
0a60: 20 73 74 72 75 63 74 20 43 6f 72 65 20 7b 0a 09   struct Core {..
0a70: 70 75 62 20 74 67 3a 20 54 67 2c 0a 09 70 75 62  pub tg: Tg,..pub
0a80: 20 64 62 3a 20 44 62 2c 0a 09 70 75 62 20 66 65   db: Db,..pub fe
0a90: 65 64 73 3a 20 41 72 63 3c 4d 75 74 65 78 3c 55  eds: Arc<Mutex<U
0aa0: 73 65 72 43 61 63 68 65 3e 3e 2c 0a 09 72 75 6e  serCache>>,..run
0ab0: 6e 69 6e 67 3a 20 41 72 63 3c 4d 75 74 65 78 3c  ning: Arc<Mutex<
0ac0: 48 61 73 68 53 65 74 3c 69 33 32 3e 3e 3e 2c 0a  HashSet<i32>>>,.
0ad0: 09 68 74 74 70 5f 63 6c 69 65 6e 74 3a 20 72 65  .http_client: re
0ae0: 71 77 65 73 74 3a 3a 43 6c 69 65 6e 74 2c 0a 7d  qwest::Client,.}
0af0: 0a 0a 70 75 62 20 73 74 72 75 63 74 20 50 6f 73  ..pub struct Pos
0b00: 74 20 7b 0a 09 75 72 69 3a 20 53 74 72 69 6e 67  t {..uri: String
0b10: 2c 0a 09 5f 74 69 74 6c 65 3a 20 53 74 72 69 6e  ,.._title: Strin
0b20: 67 2c 0a 09 5f 61 75 74 68 6f 72 73 3a 20 53 74  g,.._authors: St
0b30: 72 69 6e 67 2c 0a 09 5f 73 75 6d 6d 61 72 79 3a  ring,.._summary:
0b40: 20 53 74 72 69 6e 67 2c 0a 7d 0a 0a 69 6d 70 6c   String,.}..impl
0b50: 20 43 6f 72 65 20 7b 0a 09 2f 2f 2f 20 43 72 65   Core {../// Cre
0b60: 61 74 65 20 61 20 43 6f 72 65 20 69 6e 73 74 61  ate a Core insta
0b70: 6e 63 65 20 66 72 6f 6d 20 63 6f 6e 66 69 67 75  nce from configu
0b80: 72 61 74 69 6f 6e 20 61 6e 64 20 73 74 61 72 74  ration and start
0b90: 20 69 74 73 20 62 61 63 6b 67 72 6f 75 6e 64 20   its background 
0ba0: 61 75 74 6f 66 65 74 63 68 20 6c 6f 6f 70 2e 0a  autofetch loop..
0bb0: 09 2f 2f 2f 0a 09 2f 2f 2f 20 54 68 65 20 70 72  .///../// The pr
0bc0: 6f 76 69 64 65 64 20 60 73 65 74 74 69 6e 67 73  ovided `settings
0bd0: 60 20 6d 75 73 74 20 69 6e 63 6c 75 64 65 3a 0a  ` must include:.
0be0: 09 2f 2f 2f 20 2d 20 60 6f 77 6e 65 72 60 20 28  ./// - `owner` (
0bf0: 69 6e 74 65 67 65 72 29 3a 20 64 65 66 61 75 6c  integer): defaul
0c00: 74 20 63 68 61 74 20 69 64 20 74 6f 20 75 73 65  t chat id to use
0c10: 20 61 73 20 74 68 65 20 6f 77 6e 65 72 2f 64 65   as the owner/de
0c20: 73 74 69 6e 61 74 69 6f 6e 2c 0a 09 2f 2f 2f 20  stination,../// 
0c30: 2d 20 60 61 70 69 5f 6b 65 79 60 20 28 73 74 72  - `api_key` (str
0c40: 69 6e 67 29 3a 20 54 65 6c 65 67 72 61 6d 20 62  ing): Telegram b
0c50: 6f 74 20 41 50 49 20 6b 65 79 2c 0a 09 2f 2f 2f  ot API key,..///
0c60: 20 2d 20 60 61 70 69 5f 67 61 74 65 77 61 79 60   - `api_gateway`
0c70: 20 28 73 74 72 69 6e 67 29 3a 20 54 65 6c 65 67   (string): Teleg
0c80: 72 61 6d 20 41 50 49 20 67 61 74 65 77 61 79 20  ram API gateway 
0c90: 68 6f 73 74 2c 0a 09 2f 2f 2f 20 2d 20 60 70 67  host,../// - `pg
0ca0: 60 20 28 73 74 72 69 6e 67 29 3a 20 50 6f 73 74  ` (string): Post
0cb0: 67 72 65 53 51 4c 20 63 6f 6e 6e 65 63 74 69 6f  greSQL connectio
0cc0: 6e 20 73 74 72 69 6e 67 2c 0a 09 2f 2f 2f 20 2d  n string,../// -
0cd0: 20 6f 70 74 69 6f 6e 61 6c 20 60 70 72 6f 78 79   optional `proxy
0ce0: 60 20 28 73 74 72 69 6e 67 29 3a 20 70 72 6f 78  ` (string): prox
0cf0: 79 20 55 52 4c 20 66 6f 72 20 74 68 65 20 48 54  y URL for the HT
0d00: 54 50 20 63 6c 69 65 6e 74 2e 0a 09 2f 2f 2f 0a  TP client...///.
0d10: 09 2f 2f 2f 20 4f 6e 20 73 75 63 63 65 73 73 20  ./// On success 
0d20: 72 65 74 75 72 6e 73 20 61 6e 20 69 6e 69 74 69  returns an initi
0d30: 61 6c 69 7a 65 64 20 60 43 6f 72 65 60 20 77 69  alized `Core` wi
0d40: 74 68 20 54 65 6c 65 67 72 61 6d 20 61 6e 64 20  th Telegram and 
0d50: 48 54 54 50 20 63 6c 69 65 6e 74 73 2c 20 64 61  HTTP clients, da
0d60: 74 61 62 61 73 65 20 63 6f 6e 6e 65 63 74 69 6f  tabase connectio
0d70: 6e 2c 0a 09 2f 2f 2f 20 61 6e 20 65 6d 70 74 79  n,../// an empty
0d80: 20 72 75 6e 6e 69 6e 67 20 73 65 74 20 66 6f 72   running set for
0d90: 20 70 65 72 2d 69 64 20 74 6f 6b 65 6e 73 2c 20   per-id tokens, 
0da0: 61 6e 64 20 61 20 73 70 61 77 6e 65 64 20 62 61  and a spawned ba
0db0: 63 6b 67 72 6f 75 6e 64 20 74 61 73 6b 20 74 68  ckground task th
0dc0: 61 74 20 70 65 72 69 6f 64 69 63 61 6c 6c 79 20  at periodically 
0dd0: 72 75 6e 73 0a 09 2f 2f 2f 20 60 61 75 74 6f 66  runs../// `autof
0de0: 65 74 63 68 60 2e 20 49 66 20 61 6e 79 20 72 65  etch`. If any re
0df0: 71 75 69 72 65 64 20 73 65 74 74 69 6e 67 20 69  quired setting i
0e00: 73 20 6d 69 73 73 69 6e 67 20 6f 72 20 69 6e 69  s missing or ini
0e10: 74 69 61 6c 69 7a 61 74 69 6f 6e 20 66 61 69 6c  tialization fail
0e20: 73 2c 20 61 6e 20 65 72 72 6f 72 20 69 73 20 72  s, an error is r
0e30: 65 74 75 72 6e 65 64 2e 0a 09 70 75 62 20 61 73  eturned...pub as
0e40: 79 6e 63 20 66 6e 20 6e 65 77 28 73 65 74 74 69  ync fn new(setti
0e50: 6e 67 73 3a 20 63 6f 6e 66 69 67 3a 3a 43 6f 6e  ngs: config::Con
0e60: 66 69 67 29 20 2d 3e 20 52 65 73 75 6c 74 3c 43  fig) -> Result<C
0e70: 6f 72 65 3e 20 7b 0a 09 09 6c 65 74 20 6d 75 74  ore> {...let mut
0e80: 20 63 6c 69 65 6e 74 20 3d 20 72 65 71 77 65 73   client = reqwes
0e90: 74 3a 3a 43 6c 69 65 6e 74 3a 3a 62 75 69 6c 64  t::Client::build
0ea0: 65 72 28 29 3b 0a 09 09 69 66 20 6c 65 74 20 4f  er();...if let O
0eb0: 6b 28 70 72 6f 78 79 29 20 3d 20 73 65 74 74 69  k(proxy) = setti
0ec0: 6e 67 73 2e 67 65 74 5f 73 74 72 69 6e 67 28 22  ngs.get_string("
0ed0: 70 72 6f 78 79 22 29 20 7b 0a 09 09 09 6c 65 74  proxy") {....let
0ee0: 20 70 72 6f 78 79 20 3d 20 72 65 71 77 65 73 74   proxy = reqwest
0ef0: 3a 3a 50 72 6f 78 79 3a 3a 61 6c 6c 28 70 72 6f  ::Proxy::all(pro
0f00: 78 79 29 2e 73 74 61 63 6b 28 29 3f 3b 0a 09 09  xy).stack()?;...
0f10: 09 63 6c 69 65 6e 74 20 3d 20 63 6c 69 65 6e 74  .client = client
0f20: 2e 70 72 6f 78 79 28 70 72 6f 78 79 29 3b 0a 09  .proxy(proxy);..
0f30: 09 7d 0a 0a 09 09 6c 65 74 20 63 6f 72 65 20 3d  .}....let core =
0f40: 20 43 6f 72 65 20 7b 0a 09 09 09 74 67 3a 20 54   Core {....tg: T
0f50: 67 3a 3a 6e 65 77 28 26 73 65 74 74 69 6e 67 73  g::new(&settings
0f60: 29 2e 61 77 61 69 74 2e 73 74 61 63 6b 28 29 3f  ).await.stack()?
0f70: 2c 0a 09 09 09 64 62 3a 20 44 62 3a 3a 6e 65 77  ,....db: Db::new
0f80: 28 26 73 65 74 74 69 6e 67 73 2e 67 65 74 5f 73  (&settings.get_s
0f90: 74 72 69 6e 67 28 22 70 67 22 29 2e 73 74 61 63  tring("pg").stac
0fa0: 6b 28 29 3f 29 3f 2c 0a 09 09 09 66 65 65 64 73  k()?)?,....feeds
0fb0: 3a 20 41 72 63 3a 3a 6e 65 77 28 4d 75 74 65 78  : Arc::new(Mutex
0fc0: 3a 3a 6e 65 77 28 54 74 6c 43 61 63 68 65 3a 3a  ::new(TtlCache::
0fd0: 6e 65 77 28 31 30 30 30 30 29 29 29 2c 0a 09 09  new(10000))),...
0fe0: 09 72 75 6e 6e 69 6e 67 3a 20 41 72 63 3a 3a 6e  .running: Arc::n
0ff0: 65 77 28 4d 75 74 65 78 3a 3a 6e 65 77 28 48 61  ew(Mutex::new(Ha
1000: 73 68 53 65 74 3a 3a 6e 65 77 28 29 29 29 2c 0a  shSet::new())),.
1010: 09 09 09 68 74 74 70 5f 63 6c 69 65 6e 74 3a 20  ...http_client: 
1020: 63 6c 69 65 6e 74 2e 62 75 69 6c 64 28 29 2e 73  client.build().s
1030: 74 61 63 6b 28 29 3f 2c 0a 09 09 7d 3b 0a 0a 09  tack()?,...};...
1040: 09 6c 65 74 20 63 6c 6f 6e 65 20 3d 20 63 6f 72  .let clone = cor
1050: 65 2e 63 6c 6f 6e 65 28 29 3b 0a 09 09 73 6d 6f  e.clone();...smo
1060: 6c 3a 3a 73 70 61 77 6e 28 43 6f 6d 70 61 74 3a  l::spawn(Compat:
1070: 3a 6e 65 77 28 61 73 79 6e 63 20 6d 6f 76 65 20  :new(async move 
1080: 7b 0a 09 09 09 6c 6f 6f 70 20 7b 0a 09 09 09 09  {....loop {.....
1090: 6c 65 74 20 64 65 6c 61 79 20 3d 20 6d 61 74 63  let delay = matc
10a0: 68 20 26 63 6c 6f 6e 65 2e 61 75 74 6f 66 65 74  h &clone.autofet
10b0: 63 68 28 29 2e 61 77 61 69 74 20 7b 0a 09 09 09  ch().await {....
10c0: 09 09 45 72 72 28 65 72 72 29 20 3d 3e 20 7b 0a  ..Err(err) => {.
10d0: 09 09 09 09 09 09 69 66 20 6c 65 74 20 45 72 72  ......if let Err
10e0: 28 65 72 72 29 20 3d 20 63 6c 6f 6e 65 2e 74 67  (err) = clone.tg
10f0: 2e 73 65 6e 64 28 4d 79 4d 65 73 73 61 67 65 3a  .send(MyMessage:
1100: 3a 68 74 6d 6c 28 66 6f 72 6d 61 74 21 28 22 f0  :html(format!("ð
1110: 9f 9b 91 20 7b 65 72 72 7d 22 29 29 29 2e 61 77  Ÿ›‘ {err}"))).aw
1120: 61 69 74 20 7b 0a 09 09 09 09 09 09 09 65 70 72  ait {........epr
1130: 69 6e 74 6c 6e 21 28 22 41 75 74 6f 66 65 74 63  intln!("Autofetc
1140: 68 20 65 72 72 6f 72 3a 20 7b 65 72 72 3a 3f 7d  h error: {err:?}
1150: 22 29 3b 0a 09 09 09 09 09 09 7d 3b 0a 09 09 09  ");.......};....
1160: 09 09 09 73 74 64 3a 3a 74 69 6d 65 3a 3a 44 75  ...std::time::Du
1170: 72 61 74 69 6f 6e 3a 3a 66 72 6f 6d 5f 73 65 63  ration::from_sec
1180: 73 28 36 30 29 0a 09 09 09 09 09 7d 2c 0a 09 09  s(60)......},...
1190: 09 09 09 4f 6b 28 74 69 6d 65 29 20 3d 3e 20 2a  ...Ok(time) => *
11a0: 74 69 6d 65 2c 0a 09 09 09 09 7d 3b 0a 09 09 09  time,.....};....
11b0: 09 54 69 6d 65 72 3a 3a 61 66 74 65 72 28 64 65  .Timer::after(de
11c0: 6c 61 79 29 2e 61 77 61 69 74 3b 0a 09 09 09 7d  lay).await;....}
11d0: 0a 09 09 7d 29 29 2e 64 65 74 61 63 68 28 29 3b  ...})).detach();
11e0: 0a 09 09 4f 6b 28 63 6f 72 65 29 0a 09 7d 0a 0a  ...Ok(core)..}..
11f0: 09 2f 2f 2f 20 46 65 74 63 68 65 73 20 74 68 65  ./// Fetches the
1200: 20 66 65 65 64 20 66 6f 72 20 61 20 73 6f 75 72   feed for a sour
1210: 63 65 2c 20 73 65 6e 64 73 20 61 6e 79 20 6e 65  ce, sends any ne
1220: 77 6c 79 20 64 69 73 63 6f 76 65 72 65 64 20 70  wly discovered p
1230: 6f 73 74 73 20 74 6f 20 74 68 65 20 61 70 70 72  osts to the appr
1240: 6f 70 72 69 61 74 65 20 63 68 61 74 2c 20 61 6e  opriate chat, an
1250: 64 20 72 65 63 6f 72 64 73 20 74 68 65 6d 20 69  d records them i
1260: 6e 20 74 68 65 20 64 61 74 61 62 61 73 65 2e 0a  n the database..
1270: 09 2f 2f 2f 0a 09 2f 2f 2f 20 54 68 69 73 20 61  .///../// This a
1280: 63 71 75 69 72 65 73 20 61 20 70 65 72 2d 73 6f  cquires a per-so
1290: 75 72 63 65 20 67 75 61 72 64 20 74 6f 20 70 72  urce guard to pr
12a0: 65 76 65 6e 74 20 63 6f 6e 63 75 72 72 65 6e 74  event concurrent
12b0: 20 63 68 65 63 6b 73 20 66 6f 72 20 74 68 65 20   checks for the 
12c0: 73 61 6d 65 20 60 69 64 60 2e 20 49 66 20 61 20  same `id`. If a 
12d0: 63 68 65 63 6b 20 69 73 20 61 6c 72 65 61 64 79  check is already
12e0: 20 72 75 6e 6e 69 6e 67 20 66 6f 72 0a 09 2f 2f   running for..//
12f0: 2f 20 74 68 65 20 67 69 76 65 6e 20 60 69 64 60  / the given `id`
1300: 2c 20 74 68 65 20 66 75 6e 63 74 69 6f 6e 20 72  , the function r
1310: 65 74 75 72 6e 73 20 61 6e 20 65 72 72 6f 72 2e  eturns an error.
1320: 20 49 66 20 60 6c 61 73 74 5f 73 63 72 61 70 65   If `last_scrape
1330: 60 20 69 73 20 70 72 6f 76 69 64 65 64 2c 20 69  ` is provided, i
1340: 74 20 69 73 20 73 65 6e 74 20 61 73 20 74 68 65  t is sent as the
1350: 20 60 49 66 2d 4d 6f 64 69 66 69 65 64 2d 53 69   `If-Modified-Si
1360: 6e 63 65 60 0a 09 2f 2f 2f 20 68 65 61 64 65 72  nce`../// header
1370: 20 74 6f 20 74 68 65 20 66 65 65 64 20 72 65 71   to the feed req
1380: 75 65 73 74 2e 20 54 68 65 20 66 75 6e 63 74 69  uest. The functi
1390: 6f 6e 20 70 61 72 73 65 73 20 52 53 53 20 6f 72  on parses RSS or
13a0: 20 41 74 6f 6d 20 66 65 65 64 73 2c 20 73 65 6e   Atom feeds, sen
13b0: 64 73 20 75 6e 73 65 65 6e 20 70 6f 73 74 20 55  ds unseen post U
13c0: 52 4c 73 20 74 6f 20 65 69 74 68 65 72 20 74 68  RLs to either th
13d0: 65 20 73 6f 75 72 63 65 27 73 0a 09 2f 2f 2f 20  e source's../// 
13e0: 63 68 61 6e 6e 65 6c 20 28 77 68 65 6e 20 60 72  channel (when `r
13f0: 65 61 6c 60 20 69 73 20 74 72 75 65 29 20 6f 72  eal` is true) or
1400: 20 74 68 65 20 73 6f 75 72 63 65 20 6f 77 6e 65   the source owne
1410: 72 20 28 77 68 65 6e 20 60 72 65 61 6c 60 20 69  r (when `real` i
1420: 73 20 66 61 6c 73 65 29 2c 20 61 6e 64 20 70 65  s false), and pe
1430: 72 73 69 73 74 73 20 70 6f 73 74 65 64 20 65 6e  rsists posted en
1440: 74 72 69 65 73 20 73 6f 20 74 68 65 79 20 61 72  tries so they ar
1450: 65 0a 09 2f 2f 2f 20 6e 6f 74 20 72 65 70 6f 73  e../// not repos
1460: 74 65 64 20 6c 61 74 65 72 2e 0a 09 2f 2f 2f 0a  ted later...///.
1470: 09 2f 2f 2f 20 50 61 72 61 6d 65 74 65 72 73 3a  ./// Parameters:
1480: 0a 09 2f 2f 2f 20 2d 20 60 69 64 60 3a 20 49 64  ../// - `id`: Id
1490: 65 6e 74 69 66 69 65 72 20 6f 66 20 74 68 65 20  entifier of the 
14a0: 73 6f 75 72 63 65 20 74 6f 20 63 68 65 63 6b 2e  source to check.
14b0: 0a 09 2f 2f 2f 20 2d 20 60 72 65 61 6c 60 3a 20  ../// - `real`: 
14c0: 57 68 65 6e 20 60 74 72 75 65 60 2c 20 73 65 6e  When `true`, sen
14d0: 64 20 70 6f 73 74 73 20 74 6f 20 74 68 65 20 73  d posts to the s
14e0: 6f 75 72 63 65 27 73 20 63 68 61 6e 6e 65 6c 3b  ource's channel;
14f0: 20 77 68 65 6e 20 60 66 61 6c 73 65 60 2c 20 73   when `false`, s
1500: 65 6e 64 20 74 6f 20 74 68 65 20 73 6f 75 72 63  end to the sourc
1510: 65 20 6f 77 6e 65 72 2e 0a 09 2f 2f 2f 20 2d 20  e owner.../// - 
1520: 60 6c 61 73 74 5f 73 63 72 61 70 65 60 3a 20 4f  `last_scrape`: O
1530: 70 74 69 6f 6e 61 6c 20 74 69 6d 65 73 74 61 6d  ptional timestam
1540: 70 20 75 73 65 64 20 74 6f 20 73 65 74 20 74 68  p used to set th
1550: 65 20 60 49 66 2d 4d 6f 64 69 66 69 65 64 2d 53  e `If-Modified-S
1560: 69 6e 63 65 60 20 68 65 61 64 65 72 20 66 6f 72  ince` header for
1570: 20 74 68 65 20 48 54 54 50 20 72 65 71 75 65 73   the HTTP reques
1580: 74 2e 0a 09 2f 2f 2f 0a 09 2f 2f 2f 20 23 20 52  t...///../// # R
1590: 65 74 75 72 6e 73 0a 09 2f 2f 2f 0a 09 2f 2f 2f  eturns..///..///
15a0: 20 60 50 6f 73 74 65 64 3a 20 4e 60 20 77 68 65   `Posted: N` whe
15b0: 72 65 20 60 4e 60 20 69 73 20 74 68 65 20 6e 75  re `N` is the nu
15c0: 6d 62 65 72 20 6f 66 20 70 6f 73 74 73 20 70 72  mber of posts pr
15d0: 6f 63 65 73 73 65 64 20 61 6e 64 20 73 65 6e 74  ocessed and sent
15e0: 2e 0a 09 70 75 62 20 61 73 79 6e 63 20 66 6e 20  ...pub async fn 
15f0: 63 68 65 63 6b 20 28 26 73 65 6c 66 2c 20 69 64  check (&self, id
1600: 3a 20 69 33 32 2c 20 72 65 61 6c 3a 20 62 6f 6f  : i32, real: boo
1610: 6c 2c 20 6c 61 73 74 5f 73 63 72 61 70 65 3a 20  l, last_scrape: 
1620: 4f 70 74 69 6f 6e 3c 44 61 74 65 54 69 6d 65 3c  Option<DateTime<
1630: 4c 6f 63 61 6c 3e 3e 29 20 2d 3e 20 52 65 73 75  Local>>) -> Resu
1640: 6c 74 3c 53 74 72 69 6e 67 3e 20 7b 0a 09 09 6c  lt<String> {...l
1650: 65 74 20 6d 75 74 20 70 6f 73 74 65 64 3a 20 69  et mut posted: i
1660: 33 32 20 3d 20 30 3b 0a 09 09 6c 65 74 20 6d 75  32 = 0;...let mu
1670: 74 20 63 6f 6e 6e 20 3d 20 73 65 6c 66 2e 64 62  t conn = self.db
1680: 2e 62 65 67 69 6e 28 29 2e 61 77 61 69 74 2e 73  .begin().await.s
1690: 74 61 63 6b 28 29 3f 3b 0a 0a 09 09 6c 65 74 20  tack()?;....let 
16a0: 5f 74 6f 6b 65 6e 20 3d 20 54 6f 6b 65 6e 3a 3a  _token = Token::
16b0: 6e 65 77 28 26 73 65 6c 66 2e 72 75 6e 6e 69 6e  new(&self.runnin
16c0: 67 2c 20 69 64 29 2e 61 77 61 69 74 2e 73 74 61  g, id).await.sta
16d0: 63 6b 28 29 3f 3b 0a 09 09 6c 65 74 20 73 6f 75  ck()?;...let sou
16e0: 72 63 65 20 3d 20 63 6f 6e 6e 2e 67 65 74 5f 73  rce = conn.get_s
16f0: 6f 75 72 63 65 28 69 64 2c 20 73 65 6c 66 2e 74  ource(id, self.t
1700: 67 2e 6f 77 6e 65 72 29 2e 61 77 61 69 74 2e 73  g.owner).await.s
1710: 74 61 63 6b 28 29 3f 3b 0a 09 09 63 6f 6e 6e 2e  tack()?;...conn.
1720: 73 65 74 5f 73 63 72 61 70 65 28 69 64 29 2e 61  set_scrape(id).a
1730: 77 61 69 74 2e 73 74 61 63 6b 28 29 3f 3b 0a 09  wait.stack()?;..
1740: 09 6c 65 74 20 64 65 73 74 69 6e 61 74 69 6f 6e  .let destination
1750: 20 3d 20 43 68 61 74 50 65 65 72 49 64 3a 3a 66   = ChatPeerId::f
1760: 72 6f 6d 28 6d 61 74 63 68 20 72 65 61 6c 20 7b  rom(match real {
1770: 0a 09 09 09 74 72 75 65 20 3d 3e 20 73 6f 75 72  ....true => sour
1780: 63 65 2e 63 68 61 6e 6e 65 6c 5f 69 64 2c 0a 09  ce.channel_id,..
1790: 09 09 66 61 6c 73 65 20 3d 3e 20 73 6f 75 72 63  ..false => sourc
17a0: 65 2e 6f 77 6e 65 72 2c 0a 09 09 7d 29 3b 0a 09  e.owner,...});..
17b0: 09 6c 65 74 20 6d 75 74 20 74 68 69 73 5f 66 65  .let mut this_fe
17c0: 74 63 68 3a 20 4f 70 74 69 6f 6e 3c 44 61 74 65  tch: Option<Date
17d0: 54 69 6d 65 3c 63 68 72 6f 6e 6f 3a 3a 46 69 78  Time<chrono::Fix
17e0: 65 64 4f 66 66 73 65 74 3e 3e 20 3d 20 4e 6f 6e  edOffset>> = Non
17f0: 65 3b 0a 09 09 6c 65 74 20 6d 75 74 20 70 6f 73  e;...let mut pos
1800: 74 73 3a 20 42 54 72 65 65 4d 61 70 3c 44 61 74  ts: BTreeMap<Dat
1810: 65 54 69 6d 65 3c 63 68 72 6f 6e 6f 3a 3a 46 69  eTime<chrono::Fi
1820: 78 65 64 4f 66 66 73 65 74 3e 2c 20 50 6f 73 74  xedOffset>, Post
1830: 3e 20 3d 20 42 54 72 65 65 4d 61 70 3a 3a 6e 65  > = BTreeMap::ne
1840: 77 28 29 3b 0a 0a 09 09 6c 65 74 20 6d 75 74 20  w();....let mut 
1850: 62 75 69 6c 64 65 72 20 3d 20 73 65 6c 66 2e 68  builder = self.h
1860: 74 74 70 5f 63 6c 69 65 6e 74 2e 67 65 74 28 26  ttp_client.get(&
1870: 73 6f 75 72 63 65 2e 75 72 6c 29 3b 0a 09 09 69  source.url);...i
1880: 66 20 6c 65 74 20 53 6f 6d 65 28 6c 61 73 74 5f  f let Some(last_
1890: 73 63 72 61 70 65 29 20 3d 20 6c 61 73 74 5f 73  scrape) = last_s
18a0: 63 72 61 70 65 20 7b 0a 09 09 09 62 75 69 6c 64  crape {....build
18b0: 65 72 20 3d 20 62 75 69 6c 64 65 72 2e 68 65 61  er = builder.hea
18c0: 64 65 72 28 4c 41 53 54 5f 4d 4f 44 49 46 49 45  der(LAST_MODIFIE
18d0: 44 2c 20 6c 61 73 74 5f 73 63 72 61 70 65 2e 74  D, last_scrape.t
18e0: 6f 5f 72 66 63 32 38 32 32 28 29 29 3b 0a 09 09  o_rfc2822());...
18f0: 7d 3b 0a 09 09 6c 65 74 20 72 65 73 70 6f 6e 73  };...let respons
1900: 65 20 3d 20 62 75 69 6c 64 65 72 2e 73 65 6e 64  e = builder.send
1910: 28 29 2e 61 77 61 69 74 2e 73 74 61 63 6b 28 29  ().await.stack()
1920: 3f 3b 0a 09 09 23 5b 63 66 67 28 64 65 62 75 67  ?;...#[cfg(debug
1930: 5f 61 73 73 65 72 74 69 6f 6e 73 29 5d 0a 09 09  _assertions)]...
1940: 7b 0a 09 09 09 75 73 65 20 72 65 71 77 65 73 74  {....use reqwest
1950: 3a 3a 68 65 61 64 65 72 3a 3a 7b 0a 09 09 09 09  ::header::{.....
1960: 43 41 43 48 45 5f 43 4f 4e 54 52 4f 4c 2c 0a 09  CACHE_CONTROL,..
1970: 09 09 09 45 58 50 49 52 45 53 2c 0a 09 09 09 7d  ...EXPIRES,....}
1980: 3b 0a 09 09 09 6c 65 74 20 68 65 61 64 65 72 73  ;....let headers
1990: 20 3d 20 72 65 73 70 6f 6e 73 65 2e 68 65 61 64   = response.head
19a0: 65 72 73 28 29 3b 0a 09 09 09 6c 65 74 20 65 78  ers();....let ex
19b0: 70 69 72 65 73 20 3d 20 68 65 61 64 65 72 73 2e  pires = headers.
19c0: 67 65 74 28 45 58 50 49 52 45 53 29 3b 0a 09 09  get(EXPIRES);...
19d0: 09 6c 65 74 20 63 61 63 68 65 20 3d 20 68 65 61  .let cache = hea
19e0: 64 65 72 73 2e 67 65 74 28 43 41 43 48 45 5f 43  ders.get(CACHE_C
19f0: 4f 4e 54 52 4f 4c 29 3b 0a 09 09 09 69 66 20 65  ONTROL);....if e
1a00: 78 70 69 72 65 73 2e 69 73 5f 73 6f 6d 65 28 29  xpires.is_some()
1a10: 20 7c 7c 20 63 61 63 68 65 2e 69 73 5f 73 6f 6d   || cache.is_som
1a20: 65 28 29 20 7b 0a 09 09 09 09 70 72 69 6e 74 6c  e() {.....printl
1a30: 6e 21 28 22 7b 7d 20 7b 7d 20 7b 3a 3f 7d 20 7b  n!("{} {} {:?} {
1a40: 3a 3f 7d 20 7b 3a 3f 7d 22 2c 20 4c 6f 63 61 6c  :?} {:?}", Local
1a50: 3a 3a 6e 6f 77 28 29 2e 74 6f 5f 72 66 63 32 38  ::now().to_rfc28
1a60: 32 32 28 29 2c 20 26 73 6f 75 72 63 65 2e 75 72  22(), &source.ur
1a70: 6c 2c 20 6c 61 73 74 5f 73 63 72 61 70 65 2c 20  l, last_scrape, 
1a80: 65 78 70 69 72 65 73 2c 20 63 61 63 68 65 29 3b  expires, cache);
1a90: 0a 09 09 09 7d 0a 09 09 7d 0a 09 09 6c 65 74 20  ....}...}...let 
1aa0: 73 74 61 74 75 73 20 3d 20 72 65 73 70 6f 6e 73  status = respons
1ab0: 65 2e 73 74 61 74 75 73 28 29 3b 0a 09 09 6c 65  e.status();...le
1ac0: 74 20 63 6f 6e 74 65 6e 74 20 3d 20 72 65 73 70  t content = resp
1ad0: 6f 6e 73 65 2e 62 79 74 65 73 28 29 2e 61 77 61  onse.bytes().awa
1ae0: 69 74 2e 73 74 61 63 6b 28 29 3f 3b 0a 09 09 6d  it.stack()?;...m
1af0: 61 74 63 68 20 72 73 73 3a 3a 43 68 61 6e 6e 65  atch rss::Channe
1b00: 6c 3a 3a 72 65 61 64 5f 66 72 6f 6d 28 26 63 6f  l::read_from(&co
1b10: 6e 74 65 6e 74 5b 2e 2e 5d 29 20 7b 0a 09 09 09  ntent[..]) {....
1b20: 4f 6b 28 66 65 65 64 29 20 3d 3e 20 7b 0a 09 09  Ok(feed) => {...
1b30: 09 09 66 6f 72 20 69 74 65 6d 20 69 6e 20 66 65  ..for item in fe
1b40: 65 64 2e 69 74 65 6d 73 28 29 20 7b 0a 09 09 09  ed.items() {....
1b50: 09 09 69 66 20 6c 65 74 20 53 6f 6d 65 28 6c 69  ..if let Some(li
1b60: 6e 6b 29 20 3d 20 69 74 65 6d 2e 6c 69 6e 6b 28  nk) = item.link(
1b70: 29 20 7b 0a 09 09 09 09 09 09 6c 65 74 20 64 61  ) {.......let da
1b80: 74 65 20 3d 20 6d 61 74 63 68 20 69 74 65 6d 2e  te = match item.
1b90: 70 75 62 5f 64 61 74 65 28 29 20 7b 0a 09 09 09  pub_date() {....
1ba0: 09 09 09 09 53 6f 6d 65 28 66 65 65 64 5f 64 61  ....Some(feed_da
1bb0: 74 65 29 20 3d 3e 20 44 61 74 65 54 69 6d 65 3a  te) => DateTime:
1bc0: 3a 70 61 72 73 65 5f 66 72 6f 6d 5f 72 66 63 32  :parse_from_rfc2
1bd0: 38 32 32 28 66 65 65 64 5f 64 61 74 65 29 2c 0a  822(feed_date),.
1be0: 09 09 09 09 09 09 09 4e 6f 6e 65 20 3d 3e 20 44  .......None => D
1bf0: 61 74 65 54 69 6d 65 3a 3a 70 61 72 73 65 5f 66  ateTime::parse_f
1c00: 72 6f 6d 5f 72 66 63 33 33 33 39 28 6d 61 74 63  rom_rfc3339(matc
1c10: 68 20 69 74 65 6d 2e 64 75 62 6c 69 6e 5f 63 6f  h item.dublin_co
1c20: 72 65 5f 65 78 74 28 29 20 7b 0a 09 09 09 09 09  re_ext() {......
1c30: 09 09 09 53 6f 6d 65 28 65 78 74 29 20 3d 3e 20  ...Some(ext) => 
1c40: 7b 0a 09 09 09 09 09 09 09 09 09 6c 65 74 20 64  {..........let d
1c50: 61 74 65 73 20 3d 20 65 78 74 2e 64 61 74 65 73  ates = ext.dates
1c60: 28 29 3b 0a 09 09 09 09 09 09 09 09 09 69 66 20  ();..........if 
1c70: 64 61 74 65 73 2e 69 73 5f 65 6d 70 74 79 28 29  dates.is_empty()
1c80: 20 7b 0a 09 09 09 09 09 09 09 09 09 09 62 61 69   {...........bai
1c90: 6c 21 28 22 46 65 65 64 20 69 74 65 6d 20 68 61  l!("Feed item ha
1ca0: 73 20 44 75 62 6c 69 6e 20 43 6f 72 65 20 65 78  s Dublin Core ex
1cb0: 74 65 6e 73 69 6f 6e 20 62 75 74 20 6e 6f 20 64  tension but no d
1cc0: 61 74 65 73 2e 22 29 0a 09 09 09 09 09 09 09 09  ates.").........
1cd0: 09 7d 20 65 6c 73 65 20 7b 0a 09 09 09 09 09 09  .} else {.......
1ce0: 09 09 09 09 26 64 61 74 65 73 5b 30 5d 0a 09 09  ....&dates[0]...
1cf0: 09 09 09 09 09 09 09 7d 0a 09 09 09 09 09 09 09  .......}........
1d00: 09 7d 2c 0a 09 09 09 09 09 09 09 09 4e 6f 6e 65  .},.........None
1d10: 20 3d 3e 20 62 61 69 6c 21 28 22 46 65 65 64 20   => bail!("Feed 
1d20: 69 74 65 6d 20 6d 69 73 73 65 73 20 70 6f 73 74  item misses post
1d30: 69 6e 67 20 64 61 74 65 2e 22 29 2c 0a 09 09 09  ing date."),....
1d40: 09 09 09 09 7d 29 2c 0a 09 09 09 09 09 09 7d 2e  ....}),.......}.
1d50: 73 74 61 63 6b 28 29 3f 3b 0a 09 09 09 09 09 09  stack()?;.......
1d60: 70 6f 73 74 73 2e 69 6e 73 65 72 74 28 64 61 74  posts.insert(dat
1d70: 65 2c 20 50 6f 73 74 7b 0a 09 09 09 09 09 09 09  e, Post{........
1d80: 75 72 69 3a 20 6c 69 6e 6b 2e 74 6f 5f 73 74 72  uri: link.to_str
1d90: 69 6e 67 28 29 2c 0a 09 09 09 09 09 09 09 5f 74  ing(),........_t
1da0: 69 74 6c 65 3a 20 69 74 65 6d 2e 74 69 74 6c 65  itle: item.title
1db0: 28 29 2e 75 6e 77 72 61 70 5f 6f 72 28 22 22 29  ().unwrap_or("")
1dc0: 2e 74 6f 5f 73 74 72 69 6e 67 28 29 2c 0a 09 09  .to_string(),...
1dd0: 09 09 09 09 09 5f 61 75 74 68 6f 72 73 3a 20 69  ....._authors: i
1de0: 74 65 6d 2e 61 75 74 68 6f 72 28 29 2e 75 6e 77  tem.author().unw
1df0: 72 61 70 5f 6f 72 28 22 22 29 2e 74 6f 5f 73 74  rap_or("").to_st
1e00: 72 69 6e 67 28 29 2c 0a 09 09 09 09 09 09 09 5f  ring(),........_
1e10: 73 75 6d 6d 61 72 79 3a 20 69 74 65 6d 2e 63 6f  summary: item.co
1e20: 6e 74 65 6e 74 28 29 2e 75 6e 77 72 61 70 5f 6f  ntent().unwrap_o
1e30: 72 28 22 22 29 2e 74 6f 5f 73 74 72 69 6e 67 28  r("").to_string(
1e40: 29 2c 0a 09 09 09 09 09 09 7d 29 3b 0a 09 09 09  ),.......});....
1e50: 09 09 7d 0a 09 09 09 09 7d 3b 0a 09 09 09 7d 2c  ..}.....};....},
1e60: 0a 09 09 09 45 72 72 28 65 72 72 29 20 3d 3e 20  ....Err(err) => 
1e70: 6d 61 74 63 68 20 65 72 72 20 7b 0a 09 09 09 09  match err {.....
1e80: 72 73 73 3a 3a 45 72 72 6f 72 3a 3a 49 6e 76 61  rss::Error::Inva
1e90: 6c 69 64 53 74 61 72 74 54 61 67 20 3d 3e 20 7b  lidStartTag => {
1ea0: 0a 09 09 09 09 09 6d 61 74 63 68 20 61 74 6f 6d  ......match atom
1eb0: 5f 73 79 6e 64 69 63 61 74 69 6f 6e 3a 3a 46 65  _syndication::Fe
1ec0: 65 64 3a 3a 72 65 61 64 5f 66 72 6f 6d 28 26 63  ed::read_from(&c
1ed0: 6f 6e 74 65 6e 74 5b 2e 2e 5d 29 20 7b 0a 09 09  ontent[..]) {...
1ee0: 09 09 09 09 4f 6b 28 66 65 65 64 29 20 3d 3e 20  ....Ok(feed) => 
1ef0: 7b 0a 09 09 09 09 09 09 09 66 6f 72 20 69 74 65  {........for ite
1f00: 6d 20 69 6e 20 66 65 65 64 2e 65 6e 74 72 69 65  m in feed.entrie
1f10: 73 28 29 20 7b 0a 09 09 09 09 09 09 09 09 6c 65  s() {.........le
1f20: 74 20 64 61 74 65 20 3d 20 69 74 65 6d 2e 70 75  t date = item.pu
1f30: 62 6c 69 73 68 65 64 28 29 0a 09 09 09 09 09 09  blished().......
1f40: 09 09 09 2e 73 74 61 63 6b 5f 65 72 72 28 22 46  ....stack_err("F
1f50: 65 65 64 20 69 74 65 6d 20 6d 69 73 73 69 6e 67  eed item missing
1f60: 20 70 75 62 6c 69 73 68 69 6e 67 20 64 61 74 65   publishing date
1f70: 2e 22 29 3f 3b 0a 09 09 09 09 09 09 09 09 6c 65  .")?;.........le
1f80: 74 20 75 72 69 20 3d 20 7b 0a 09 09 09 09 09 09  t uri = {.......
1f90: 09 09 09 6c 65 74 20 6c 69 6e 6b 73 20 3d 20 69  ...let links = i
1fa0: 74 65 6d 2e 6c 69 6e 6b 73 28 29 3b 0a 09 09 09  tem.links();....
1fb0: 09 09 09 09 09 09 69 66 20 6c 69 6e 6b 73 2e 69  ......if links.i
1fc0: 73 5f 65 6d 70 74 79 28 29 20 7b 0a 09 09 09 09  s_empty() {.....
1fd0: 09 09 09 09 09 09 62 61 69 6c 21 28 22 46 65 65  ......bail!("Fee
1fe0: 64 20 69 74 65 6d 20 6d 69 73 73 69 6e 67 20 70  d item missing p
1ff0: 6f 73 74 20 6c 69 6e 6b 73 2e 22 29 3b 0a 09 09  ost links.");...
2000: 09 09 09 09 09 09 09 7d 20 65 6c 73 65 20 7b 0a  .......} else {.
2010: 09 09 09 09 09 09 09 09 09 09 6c 69 6e 6b 73 5b  ..........links[
2020: 30 5d 2e 68 72 65 66 28 29 2e 74 6f 5f 73 74 72  0].href().to_str
2030: 69 6e 67 28 29 0a 09 09 09 09 09 09 09 09 09 7d  ing()..........}
2040: 0a 09 09 09 09 09 09 09 09 7d 3b 0a 09 09 09 09  .........};.....
2050: 09 09 09 09 6c 65 74 20 5f 61 75 74 68 6f 72 73  ....let _authors
2060: 20 3d 20 69 74 65 6d 2e 61 75 74 68 6f 72 73 28   = item.authors(
2070: 29 2e 69 74 65 72 28 29 2e 6d 61 70 28 7c 78 7c  ).iter().map(|x|
2080: 20 66 6f 72 6d 61 74 21 28 22 7b 7d 20 3c 7b 3a   format!("{} <{:
2090: 3f 7d 3e 22 2c 20 78 2e 6e 61 6d 65 28 29 2c 20  ?}>", x.name(), 
20a0: 78 2e 65 6d 61 69 6c 28 29 29 29 2e 63 6f 6c 6c  x.email())).coll
20b0: 65 63 74 3a 3a 3c 56 65 63 3c 53 74 72 69 6e 67  ect::<Vec<String
20c0: 3e 3e 28 29 2e 6a 6f 69 6e 28 22 2c 20 22 29 3b  >>().join(", ");
20d0: 0a 09 09 09 09 09 09 09 09 6c 65 74 20 5f 73 75  .........let _su
20e0: 6d 6d 61 72 79 20 3d 20 69 66 20 6c 65 74 20 53  mmary = if let S
20f0: 6f 6d 65 28 73 75 6d 29 20 3d 20 69 74 65 6d 2e  ome(sum) = item.
2100: 73 75 6d 6d 61 72 79 28 29 20 7b 20 73 75 6d 2e  summary() { sum.
2110: 76 61 6c 75 65 2e 63 6c 6f 6e 65 28 29 20 7d 20  value.clone() } 
2120: 65 6c 73 65 20 7b 20 53 74 72 69 6e 67 3a 3a 6e  else { String::n
2130: 65 77 28 29 20 7d 3b 0a 09 09 09 09 09 09 09 09  ew() };.........
2140: 70 6f 73 74 73 2e 69 6e 73 65 72 74 28 2a 64 61  posts.insert(*da
2150: 74 65 2c 20 50 6f 73 74 7b 0a 09 09 09 09 09 09  te, Post{.......
2160: 09 09 09 75 72 69 2c 0a 09 09 09 09 09 09 09 09  ...uri,.........
2170: 09 5f 74 69 74 6c 65 3a 20 69 74 65 6d 2e 74 69  ._title: item.ti
2180: 74 6c 65 28 29 2e 74 6f 5f 73 74 72 69 6e 67 28  tle().to_string(
2190: 29 2c 0a 09 09 09 09 09 09 09 09 09 5f 61 75 74  ),.........._aut
21a0: 68 6f 72 73 2c 0a 09 09 09 09 09 09 09 09 09 5f  hors,.........._
21b0: 73 75 6d 6d 61 72 79 2c 0a 09 09 09 09 09 09 09  summary,........
21c0: 09 7d 29 3b 0a 09 09 09 09 09 09 09 7d 3b 0a 09  .});........};..
21d0: 09 09 09 09 09 7d 2c 0a 09 09 09 09 09 09 45 72  .....},.......Er
21e0: 72 28 65 72 72 29 20 3d 3e 20 7b 0a 09 09 09 09  r(err) => {.....
21f0: 09 09 09 62 61 69 6c 21 28 22 55 6e 73 75 70 70  ...bail!("Unsupp
2200: 6f 72 74 65 64 20 6f 72 20 6d 61 6e 67 6c 65 64  orted or mangled
2210: 20 63 6f 6e 74 65 6e 74 3a 5c 6e 7b 3a 3f 7d 5c   content:\n{:?}\
2220: 6e 7b 65 72 72 7d 5c 6e 7b 73 74 61 74 75 73 3a  n{err}\n{status:
2230: 23 3f 7d 5c 6e 22 2c 20 26 73 6f 75 72 63 65 2e  #?}\n", &source.
2240: 75 72 6c 29 0a 09 09 09 09 09 09 7d 2c 0a 09 09  url).......},...
2250: 09 09 09 7d 0a 09 09 09 09 7d 2c 0a 09 09 09 09  ...}.....},.....
2260: 72 73 73 3a 3a 45 72 72 6f 72 3a 3a 45 6f 66 20  rss::Error::Eof 
2270: 3d 3e 20 28 29 2c 0a 09 09 09 09 5f 20 3d 3e 20  => (),....._ => 
2280: 62 61 69 6c 21 28 22 55 6e 73 75 70 70 6f 72 74  bail!("Unsupport
2290: 65 64 20 6f 72 20 6d 61 6e 67 6c 65 64 20 63 6f  ed or mangled co
22a0: 6e 74 65 6e 74 3a 5c 6e 7b 3a 3f 7d 5c 6e 7b 65  ntent:\n{:?}\n{e
22b0: 72 72 7d 5c 6e 7b 73 74 61 74 75 73 3a 23 3f 7d  rr}\n{status:#?}
22c0: 5c 6e 22 2c 20 26 73 6f 75 72 63 65 2e 75 72 6c  \n", &source.url
22d0: 29 0a 09 09 09 7d 0a 09 09 7d 3b 0a 09 09 66 6f  )....}...};...fo
22e0: 72 20 28 64 61 74 65 2c 20 70 6f 73 74 29 20 69  r (date, post) i
22f0: 6e 20 70 6f 73 74 73 2e 69 74 65 72 28 29 20 7b  n posts.iter() {
2300: 0a 09 09 09 6c 65 74 20 70 6f 73 74 5f 75 72 6c  ....let post_url
2310: 3a 20 43 6f 77 3c 73 74 72 3e 20 3d 20 6d 61 74  : Cow<str> = mat
2320: 63 68 20 73 6f 75 72 63 65 2e 75 72 6c 5f 72 65  ch source.url_re
2330: 20 7b 0a 09 09 09 09 53 6f 6d 65 28 72 65 66 20   {.....Some(ref 
2340: 78 29 20 3d 3e 20 73 65 64 72 65 67 65 78 3a 3a  x) => sedregex::
2350: 52 65 70 6c 61 63 65 43 6f 6d 6d 61 6e 64 3a 3a  ReplaceCommand::
2360: 6e 65 77 28 78 29 2e 73 74 61 63 6b 28 29 3f 2e  new(x).stack()?.
2370: 65 78 65 63 75 74 65 28 26 70 6f 73 74 2e 75 72  execute(&post.ur
2380: 69 29 2c 0a 09 09 09 09 4e 6f 6e 65 20 3d 3e 20  i),.....None => 
2390: 70 6f 73 74 2e 75 72 69 2e 63 6c 6f 6e 65 28 29  post.uri.clone()
23a0: 2e 69 6e 74 6f 28 29 2c 0a 09 09 09 7d 3b 0a 09  .into(),....};..
23b0: 09 09 69 66 20 21 20 63 6f 6e 6e 2e 65 78 69 73  ..if ! conn.exis
23c0: 74 73 28 26 70 6f 73 74 5f 75 72 6c 2c 20 69 64  ts(&post_url, id
23d0: 29 2e 61 77 61 69 74 2e 73 74 61 63 6b 28 29 3f  ).await.stack()?
23e0: 20 7b 0a 09 09 09 09 69 66 20 74 68 69 73 5f 66   {.....if this_f
23f0: 65 74 63 68 2e 69 73 5f 6e 6f 6e 65 28 29 20 7c  etch.is_none() |
2400: 7c 20 2a 64 61 74 65 20 3e 20 74 68 69 73 5f 66  | *date > this_f
2410: 65 74 63 68 2e 75 6e 77 72 61 70 28 29 20 7b 0a  etch.unwrap() {.
2420: 09 09 09 09 09 74 68 69 73 5f 66 65 74 63 68 20  .....this_fetch 
2430: 3d 20 53 6f 6d 65 28 2a 64 61 74 65 29 3b 0a 09  = Some(*date);..
2440: 09 09 09 7d 3b 0a 09 09 09 09 73 65 6c 66 2e 74  ...};.....self.t
2450: 67 2e 73 65 6e 64 28 4d 79 4d 65 73 73 61 67 65  g.send(MyMessage
2460: 3a 3a 68 74 6d 6c 5f 74 6f 28 6d 61 74 63 68 20  ::html_to(match 
2470: 26 73 6f 75 72 63 65 2e 69 76 5f 68 61 73 68 20  &source.iv_hash 
2480: 7b 0a 09 09 09 09 09 53 6f 6d 65 28 68 61 73 68  {......Some(hash
2490: 29 20 3d 3e 20 66 6f 72 6d 61 74 21 28 22 3c 61  ) => format!("<a
24a0: 20 68 72 65 66 3d 5c 22 68 74 74 70 73 3a 2f 2f   href=\"https://
24b0: 74 2e 6d 65 2f 69 76 3f 75 72 6c 3d 7b 70 6f 73  t.me/iv?url={pos
24c0: 74 5f 75 72 6c 7d 26 72 68 61 73 68 3d 7b 68 61  t_url}&rhash={ha
24d0: 73 68 7d 5c 22 3e 20 3c 2f 61 3e 7b 70 6f 73 74  sh}\"> </a>{post
24e0: 5f 75 72 6c 7d 22 29 2c 0a 09 09 09 09 09 4e 6f  _url}"),......No
24f0: 6e 65 20 3d 3e 20 66 6f 72 6d 61 74 21 28 22 7b  ne => format!("{
2500: 70 6f 73 74 5f 75 72 6c 7d 22 29 2c 0a 09 09 09  post_url}"),....
2510: 09 7d 2c 20 64 65 73 74 69 6e 61 74 69 6f 6e 29  .}, destination)
2520: 29 2e 61 77 61 69 74 2e 73 74 61 63 6b 28 29 3f  ).await.stack()?
2530: 3b 0a 09 09 09 09 63 6f 6e 6e 2e 61 64 64 5f 70  ;.....conn.add_p
2540: 6f 73 74 28 69 64 2c 20 64 61 74 65 2c 20 26 70  ost(id, date, &p
2550: 6f 73 74 5f 75 72 6c 29 2e 61 77 61 69 74 2e 73  ost_url).await.s
2560: 74 61 63 6b 28 29 3f 3b 0a 09 09 09 09 70 6f 73  tack()?;.....pos
2570: 74 65 64 20 2b 3d 20 31 3b 0a 09 09 09 7d 3b 0a  ted += 1;....};.
2580: 09 09 7d 3b 0a 09 09 70 6f 73 74 73 2e 63 6c 65  ..};...posts.cle
2590: 61 72 28 29 3b 0a 09 09 4f 6b 28 66 6f 72 6d 61  ar();...Ok(forma
25a0: 74 21 28 22 50 6f 73 74 65 64 3a 20 7b 70 6f 73  t!("Posted: {pos
25b0: 74 65 64 7d 22 29 29 0a 09 7d 0a 0a 09 2f 2f 2f  ted}"))..}...///
25c0: 20 44 65 74 65 72 6d 69 6e 65 20 74 68 65 20 64   Determine the d
25d0: 65 6c 61 79 20 75 6e 74 69 6c 20 74 68 65 20 6e  elay until the n
25e0: 65 78 74 20 73 63 68 65 64 75 6c 65 64 20 66 65  ext scheduled fe
25f0: 74 63 68 20 61 6e 64 20 73 70 61 77 6e 20 62 61  tch and spawn ba
2600: 63 6b 67 72 6f 75 6e 64 20 63 68 65 63 6b 73 20  ckground checks 
2610: 66 6f 72 20 61 6e 79 20 6f 76 65 72 64 75 65 20  for any overdue 
2620: 73 6f 75 72 63 65 73 2e 0a 09 2f 2f 2f 0a 09 2f  sources...///../
2630: 2f 2f 20 54 68 69 73 20 73 63 61 6e 73 20 74 68  // This scans th
2640: 65 20 64 61 74 61 62 61 73 65 20 71 75 65 75 65  e database queue
2650: 2c 20 73 70 61 77 6e 73 20 62 61 63 6b 67 72 6f  , spawns backgro
2660: 75 6e 64 20 74 61 73 6b 73 20 74 6f 20 72 75 6e  und tasks to run
2670: 20 63 68 65 63 6b 73 20 66 6f 72 20 73 6f 75 72   checks for sour
2680: 63 65 73 20 77 68 6f 73 65 20 60 6e 65 78 74 5f  ces whose `next_
2690: 66 65 74 63 68 60 0a 09 2f 2f 2f 20 69 73 20 69  fetch`../// is i
26a0: 6e 20 74 68 65 20 70 61 73 74 20 28 65 61 63 68  n the past (each
26b0: 20 74 61 73 6b 20 75 73 65 73 20 61 20 43 6f 72   task uses a Cor
26c0: 65 20 63 6c 6f 6e 65 20 77 69 74 68 20 74 68 65  e clone with the
26d0: 20 61 70 70 72 6f 70 72 69 61 74 65 20 6f 77 6e   appropriate own
26e0: 65 72 29 2c 20 61 6e 64 20 63 6f 6d 70 75 74 65  er), and compute
26f0: 73 20 74 68 65 20 73 68 6f 72 74 65 73 74 0a 09  s the shortest..
2700: 2f 2f 2f 20 64 75 72 61 74 69 6f 6e 20 75 6e 74  /// duration unt
2710: 69 6c 20 74 68 65 20 6e 65 78 74 20 60 6e 65 78  il the next `nex
2720: 74 5f 66 65 74 63 68 60 2e 0a 09 61 73 79 6e 63  t_fetch`...async
2730: 20 66 6e 20 61 75 74 6f 66 65 74 63 68 28 26 73   fn autofetch(&s
2740: 65 6c 66 29 20 2d 3e 20 52 65 73 75 6c 74 3c 73  elf) -> Result<s
2750: 74 64 3a 3a 74 69 6d 65 3a 3a 44 75 72 61 74 69  td::time::Durati
2760: 6f 6e 3e 20 7b 0a 09 09 6c 65 74 20 6d 75 74 20  on> {...let mut 
2770: 64 65 6c 61 79 20 3d 20 63 68 72 6f 6e 6f 3a 3a  delay = chrono::
2780: 44 75 72 61 74 69 6f 6e 3a 3a 6d 69 6e 75 74 65  Duration::minute
2790: 73 28 31 29 3b 0a 09 09 6c 65 74 20 6e 6f 77 20  s(1);...let now 
27a0: 3d 20 63 68 72 6f 6e 6f 3a 3a 4c 6f 63 61 6c 3a  = chrono::Local:
27b0: 3a 6e 6f 77 28 29 3b 0a 09 09 6c 65 74 20 71 75  :now();...let qu
27c0: 65 75 65 20 3d 20 7b 0a 09 09 09 6c 65 74 20 6d  eue = {....let m
27d0: 75 74 20 63 6f 6e 6e 20 3d 20 73 65 6c 66 2e 64  ut conn = self.d
27e0: 62 2e 62 65 67 69 6e 28 29 2e 61 77 61 69 74 2e  b.begin().await.
27f0: 73 74 61 63 6b 28 29 3f 3b 0a 09 09 09 63 6f 6e  stack()?;....con
2800: 6e 2e 67 65 74 5f 71 75 65 75 65 28 29 2e 61 77  n.get_queue().aw
2810: 61 69 74 2e 73 74 61 63 6b 28 29 3f 0a 09 09 7d  ait.stack()?...}
2820: 3b 0a 09 09 66 6f 72 20 72 6f 77 20 69 6e 20 71  ;...for row in q
2830: 75 65 75 65 20 7b 0a 09 09 09 69 66 20 6c 65 74  ueue {....if let
2840: 20 53 6f 6d 65 28 6e 65 78 74 5f 66 65 74 63 68   Some(next_fetch
2850: 29 20 3d 20 72 6f 77 2e 6e 65 78 74 5f 66 65 74  ) = row.next_fet
2860: 63 68 20 7b 0a 09 09 09 09 69 66 20 6e 65 78 74  ch {.....if next
2870: 5f 66 65 74 63 68 20 3c 20 6e 6f 77 20 7b 0a 09  _fetch < now {..
2880: 09 09 09 09 69 66 20 6c 65 74 20 28 53 6f 6d 65  ....if let (Some
2890: 28 6f 77 6e 65 72 29 2c 20 53 6f 6d 65 28 73 6f  (owner), Some(so
28a0: 75 72 63 65 5f 69 64 29 2c 20 6c 61 73 74 5f 73  urce_id), last_s
28b0: 63 72 61 70 65 29 20 3d 20 28 72 6f 77 2e 6f 77  crape) = (row.ow
28c0: 6e 65 72 2c 20 72 6f 77 2e 73 6f 75 72 63 65 5f  ner, row.source_
28d0: 69 64 2c 20 72 6f 77 2e 6c 61 73 74 5f 73 63 72  id, row.last_scr
28e0: 61 70 65 29 20 7b 0a 09 09 09 09 09 09 6c 65 74  ape) {.......let
28f0: 20 63 6c 6f 6e 65 20 3d 20 43 6f 72 65 20 7b 0a   clone = Core {.
2900: 09 09 09 09 09 09 09 74 67 3a 20 73 65 6c 66 2e  .......tg: self.
2910: 74 67 2e 77 69 74 68 5f 6f 77 6e 65 72 28 6f 77  tg.with_owner(ow
2920: 6e 65 72 29 2c 0a 09 09 09 09 09 09 09 2e 2e 73  ner),..........s
2930: 65 6c 66 2e 63 6c 6f 6e 65 28 29 0a 09 09 09 09  elf.clone().....
2940: 09 09 7d 3b 0a 09 09 09 09 09 09 6c 65 74 20 73  ..};.......let s
2950: 6f 75 72 63 65 20 3d 20 7b 0a 09 09 09 09 09 09  ource = {.......
2960: 09 6c 65 74 20 6d 75 74 20 63 6f 6e 6e 20 3d 20  .let mut conn = 
2970: 73 65 6c 66 2e 64 62 2e 62 65 67 69 6e 28 29 2e  self.db.begin().
2980: 61 77 61 69 74 2e 73 74 61 63 6b 28 29 3f 3b 0a  await.stack()?;.
2990: 09 09 09 09 09 09 09 6d 61 74 63 68 20 63 6f 6e  .......match con
29a0: 6e 2e 67 65 74 5f 6f 6e 65 28 6f 77 6e 65 72 2c  n.get_one(owner,
29b0: 20 73 6f 75 72 63 65 5f 69 64 29 2e 61 77 61 69   source_id).awai
29c0: 74 20 7b 0a 09 09 09 09 09 09 09 09 4f 6b 28 53  t {.........Ok(S
29d0: 6f 6d 65 28 73 6f 75 72 63 65 29 29 20 3d 3e 20  ome(source)) => 
29e0: 73 6f 75 72 63 65 2e 74 6f 5f 73 74 72 69 6e 67  source.to_string
29f0: 28 29 2c 0a 09 09 09 09 09 09 09 09 4f 6b 28 4e  (),.........Ok(N
2a00: 6f 6e 65 29 20 3d 3e 20 22 53 6f 75 72 63 65 20  one) => "Source 
2a10: 6e 6f 74 20 66 6f 75 6e 64 20 69 6e 20 64 61 74  not found in dat
2a20: 61 62 61 73 65 3f 22 2e 74 6f 5f 73 74 72 69 6e  abase?".to_strin
2a30: 67 28 29 2c 0a 09 09 09 09 09 09 09 09 45 72 72  g(),.........Err
2a40: 28 65 72 72 29 20 3d 3e 20 66 6f 72 6d 61 74 21  (err) => format!
2a50: 28 22 46 61 69 6c 65 64 20 74 6f 20 66 65 74 63  ("Failed to fetc
2a60: 68 20 73 6f 75 72 63 65 20 64 61 74 61 3a 5c 6e  h source data:\n
2a70: 7b 65 72 72 7d 22 29 2c 0a 09 09 09 09 09 09 09  {err}"),........
2a80: 7d 0a 09 09 09 09 09 09 7d 3b 0a 09 09 09 09 09  }.......};......
2a90: 09 73 6d 6f 6c 3a 3a 73 70 61 77 6e 28 43 6f 6d  .smol::spawn(Com
2aa0: 70 61 74 3a 3a 6e 65 77 28 61 73 79 6e 63 20 6d  pat::new(async m
2ab0: 6f 76 65 20 7b 0a 09 09 09 09 09 09 09 69 66 20  ove {........if 
2ac0: 6c 65 74 20 45 72 72 28 65 72 72 29 20 3d 20 63  let Err(err) = c
2ad0: 6c 6f 6e 65 2e 63 68 65 63 6b 28 73 6f 75 72 63  lone.check(sourc
2ae0: 65 5f 69 64 2c 20 74 72 75 65 2c 20 53 6f 6d 65  e_id, true, Some
2af0: 28 6c 61 73 74 5f 73 63 72 61 70 65 29 29 2e 61  (last_scrape)).a
2b00: 77 61 69 74 0a 09 09 09 09 09 09 09 09 26 26 20  wait.........&& 
2b10: 6c 65 74 20 45 72 72 28 65 72 72 29 20 3d 20 63  let Err(err) = c
2b20: 6c 6f 6e 65 2e 74 67 2e 73 65 6e 64 28 4d 79 4d  lone.tg.send(MyM
2b30: 65 73 73 61 67 65 3a 3a 74 65 78 74 28 66 6f 72  essage::text(for
2b40: 6d 61 74 21 28 22 f0 9f 9b 91 20 7b 73 6f 75 72  mat!("🛑 {sour
2b50: 63 65 7d 5c 6e 7b 7d 22 2c 20 65 6e 63 6f 64 65  ce}\n{}", encode
2b60: 28 26 65 72 72 2e 74 6f 5f 73 74 72 69 6e 67 28  (&err.to_string(
2b70: 29 29 29 29 29 2e 61 77 61 69 74 0a 09 09 09 09  ))))).await.....
2b80: 09 09 09 7b 0a 09 09 09 09 09 09 09 09 65 70 72  ...{.........epr
2b90: 69 6e 74 6c 6e 21 28 22 43 68 65 63 6b 20 65 72  intln!("Check er
2ba0: 72 6f 72 3a 20 7b 65 72 72 7d 22 29 3b 0a 09 09  ror: {err}");...
2bb0: 09 09 09 09 09 7d 3b 0a 09 09 09 09 09 09 7d 29  .....};.......})
2bc0: 29 2e 64 65 74 61 63 68 28 29 3b 0a 09 09 09 09  ).detach();.....
2bd0: 09 7d 0a 09 09 09 09 7d 20 65 6c 73 65 20 69 66  .}.....} else if
2be0: 20 6e 65 78 74 5f 66 65 74 63 68 20 2d 20 6e 6f   next_fetch - no
2bf0: 77 20 3c 20 64 65 6c 61 79 20 7b 0a 09 09 09 09  w < delay {.....
2c00: 09 64 65 6c 61 79 20 3d 20 6e 65 78 74 5f 66 65  .delay = next_fe
2c10: 74 63 68 20 2d 20 6e 6f 77 3b 0a 09 09 09 09 7d  tch - now;.....}
2c20: 0a 09 09 09 7d 0a 09 09 7d 3b 0a 09 09 64 65 6c  ....}...};...del
2c30: 61 79 2e 74 6f 5f 73 74 64 28 29 2e 73 74 61 63  ay.to_std().stac
2c40: 6b 28 29 0a 09 7d 0a 0a 09 2f 2f 2f 20 44 69 73  k()..}.../// Dis
2c50: 70 6c 61 79 73 20 66 75 6c 6c 20 6c 69 73 74 20  plays full list 
2c60: 6f 66 20 6d 61 6e 61 67 65 64 20 63 68 61 6e 6e  of managed chann
2c70: 65 6c 73 20 66 6f 72 20 73 70 65 63 69 66 69 65  els for specifie
2c80: 64 20 75 73 65 72 0a 09 70 75 62 20 61 73 79 6e  d user..pub asyn
2c90: 63 20 66 6e 20 6c 69 73 74 20 28 26 73 65 6c 66  c fn list (&self
2ca0: 2c 20 6f 77 6e 65 72 3a 20 55 73 65 72 50 65 65  , owner: UserPee
2cb0: 72 49 64 29 20 2d 3e 20 52 65 73 75 6c 74 3c 53  rId) -> Result<S
2cc0: 74 72 69 6e 67 3e 20 7b 0a 09 09 6c 65 74 20 6d  tring> {...let m
2cd0: 75 74 20 72 65 70 6c 79 3a 20 56 65 63 3c 53 74  ut reply: Vec<St
2ce0: 72 69 6e 67 3e 20 3d 20 76 65 63 21 5b 5d 3b 0a  ring> = vec![];.
2cf0: 09 09 72 65 70 6c 79 2e 70 75 73 68 28 22 43 68  ..reply.push("Ch
2d00: 61 6e 6e 65 6c 73 3a 22 2e 69 6e 74 6f 28 29 29  annels:".into())
2d10: 3b 0a 09 09 6c 65 74 20 6d 75 74 20 63 6f 6e 6e  ;...let mut conn
2d20: 20 3d 20 73 65 6c 66 2e 64 62 2e 62 65 67 69 6e   = self.db.begin
2d30: 28 29 2e 61 77 61 69 74 2e 73 74 61 63 6b 28 29  ().await.stack()
2d40: 3f 3b 0a 09 09 66 6f 72 20 72 6f 77 20 69 6e 20  ?;...for row in 
2d50: 63 6f 6e 6e 2e 67 65 74 5f 6c 69 73 74 28 6f 77  conn.get_list(ow
2d60: 6e 65 72 29 2e 61 77 61 69 74 2e 73 74 61 63 6b  ner).await.stack
2d70: 28 29 3f 20 7b 0a 09 09 09 72 65 70 6c 79 2e 70  ()? {....reply.p
2d80: 75 73 68 28 72 6f 77 2e 74 6f 5f 73 74 72 69 6e  ush(row.to_strin
2d90: 67 28 29 29 3b 0a 09 09 7d 3b 0a 09 09 4f 6b 28  g());...};...Ok(
2da0: 72 65 70 6c 79 2e 6a 6f 69 6e 28 22 5c 6e 5c 6e  reply.join("\n\n
2db0: 22 29 29 0a 09 7d 0a 0a 09 2f 2f 2f 20 52 65 74  "))..}.../// Ret
2dc0: 75 72 6e 73 20 63 75 72 72 65 6e 74 20 63 61 63  urns current cac
2dd0: 68 65 64 20 6c 69 73 74 20 6f 66 20 66 65 65 64  hed list of feed
2de0: 20 66 6f 72 20 72 65 71 75 65 73 74 65 64 20 75   for requested u
2df0: 73 65 72 2c 20 6f 72 20 6c 6f 61 64 73 20 64 61  ser, or loads da
2e00: 74 61 20 66 72 6f 6d 20 64 61 74 61 62 61 73 65  ta from database
2e10: 0a 09 70 75 62 20 61 73 79 6e 63 20 66 6e 20 67  ..pub async fn g
2e20: 65 74 5f 66 65 65 64 73 20 28 26 73 65 6c 66 2c  et_feeds (&self,
2e30: 20 6f 77 6e 65 72 3a 20 69 36 34 29 20 2d 3e 20   owner: i64) -> 
2e40: 52 65 73 75 6c 74 3c 41 72 63 3c 4d 75 74 65 78  Result<Arc<Mutex
2e50: 3c 48 61 73 68 4d 61 70 3c 69 33 32 2c 20 53 74  <HashMap<i32, St
2e60: 72 69 6e 67 3e 3e 3e 3e 20 7b 0a 09 09 6c 65 74  ring>>>> {...let
2e70: 20 6d 75 74 20 66 65 65 64 73 20 3d 20 73 65 6c   mut feeds = sel
2e80: 66 2e 66 65 65 64 73 2e 6c 6f 63 6b 5f 61 72 63  f.feeds.lock_arc
2e90: 28 29 2e 61 77 61 69 74 3b 0a 09 09 4f 6b 28 6d  ().await;...Ok(m
2ea0: 61 74 63 68 20 66 65 65 64 73 2e 67 65 74 28 26  atch feeds.get(&
2eb0: 6f 77 6e 65 72 29 20 7b 0a 09 09 09 4e 6f 6e 65  owner) {....None
2ec0: 20 3d 3e 20 7b 0a 09 09 09 09 6c 65 74 20 6d 75   => {.....let mu
2ed0: 74 20 63 6f 6e 6e 20 3d 20 73 65 6c 66 2e 64 62  t conn = self.db
2ee0: 2e 62 65 67 69 6e 28 29 2e 61 77 61 69 74 2e 73  .begin().await.s
2ef0: 74 61 63 6b 28 29 3f 3b 0a 09 09 09 09 6c 65 74  tack()?;.....let
2f00: 20 66 65 65 64 5f 6c 69 73 74 20 3d 20 63 6f 6e   feed_list = con
2f10: 6e 2e 67 65 74 5f 66 65 65 64 73 28 6f 77 6e 65  n.get_feeds(owne
2f20: 72 29 2e 61 77 61 69 74 2e 73 74 61 63 6b 28 29  r).await.stack()
2f30: 3f 3b 0a 09 09 09 09 6c 65 74 20 6d 75 74 20 6d  ?;.....let mut m
2f40: 61 70 20 3d 20 48 61 73 68 4d 61 70 3a 3a 6e 65  ap = HashMap::ne
2f50: 77 28 29 3b 0a 09 09 09 09 66 6f 72 20 66 65 65  w();.....for fee
2f60: 64 20 69 6e 20 66 65 65 64 5f 6c 69 73 74 20 7b  d in feed_list {
2f70: 0a 09 09 09 09 09 6d 61 70 2e 69 6e 73 65 72 74  ......map.insert
2f80: 28 66 65 65 64 2e 73 6f 75 72 63 65 5f 69 64 2c  (feed.source_id,
2f90: 20 66 65 65 64 2e 63 68 61 6e 6e 65 6c 29 3b 0a   feed.channel);.
2fa0: 09 09 09 09 7d 3b 0a 09 09 09 09 6c 65 74 20 72  ....};.....let r
2fb0: 65 73 20 3d 20 41 72 63 3a 3a 6e 65 77 28 4d 75  es = Arc::new(Mu
2fc0: 74 65 78 3a 3a 6e 65 77 28 6d 61 70 29 29 3b 0a  tex::new(map));.
2fd0: 09 09 09 09 66 65 65 64 73 2e 69 6e 73 65 72 74  ....feeds.insert
2fe0: 28 6f 77 6e 65 72 2c 20 72 65 73 2e 63 6c 6f 6e  (owner, res.clon
2ff0: 65 28 29 2c 20 44 75 72 61 74 69 6f 6e 3a 3a 66  e(), Duration::f
3000: 72 6f 6d 5f 73 65 63 73 28 36 30 20 2a 20 36 30  rom_secs(60 * 60
3010: 20 2a 20 33 29 29 3b 0a 09 09 09 09 72 65 73 0a   * 3));.....res.
3020: 09 09 09 7d 2c 0a 09 09 09 53 6f 6d 65 28 72 65  ...},....Some(re
3030: 73 29 20 3d 3e 20 72 65 73 2e 63 6c 6f 6e 65 28  s) => res.clone(
3040: 29 2c 0a 09 09 7d 29 0a 09 7d 0a 0a 09 2f 2f 2f  ),...})..}...///
3050: 20 41 64 64 73 20 66 65 65 64 20 74 6f 20 63 61   Adds feed to ca
3060: 63 68 65 64 20 6c 69 73 74 0a 09 70 75 62 20 61  ched list..pub a
3070: 73 79 6e 63 20 66 6e 20 61 64 64 5f 66 65 65 64  sync fn add_feed
3080: 20 28 26 73 65 6c 66 2c 20 6f 77 6e 65 72 3a 20   (&self, owner: 
3090: 69 36 34 2c 20 73 6f 75 72 63 65 5f 69 64 3a 20  i64, source_id: 
30a0: 69 33 32 2c 20 63 68 61 6e 6e 65 6c 3a 20 53 74  i32, channel: St
30b0: 72 69 6e 67 29 20 2d 3e 20 52 65 73 75 6c 74 3c  ring) -> Result<
30c0: 28 29 3e 20 7b 0a 09 09 6c 65 74 20 6d 75 74 20  ()> {...let mut 
30d0: 69 6e 73 65 72 74 65 64 20 3d 20 74 72 75 65 3b  inserted = true;
30e0: 0a 09 09 7b 0a 09 09 09 6c 65 74 20 6d 75 74 20  ...{....let mut 
30f0: 66 65 65 64 73 20 3d 20 73 65 6c 66 2e 66 65 65  feeds = self.fee
3100: 64 73 2e 6c 6f 63 6b 5f 61 72 63 28 29 2e 61 77  ds.lock_arc().aw
3110: 61 69 74 3b 0a 09 09 09 69 66 20 6c 65 74 20 53  ait;....if let S
3120: 6f 6d 65 28 66 65 65 64 29 20 3d 20 66 65 65 64  ome(feed) = feed
3130: 73 2e 67 65 74 5f 6d 75 74 28 26 6f 77 6e 65 72  s.get_mut(&owner
3140: 29 20 7b 0a 09 09 09 09 6c 65 74 20 6d 75 74 20  ) {.....let mut 
3150: 66 65 65 64 20 3d 20 66 65 65 64 2e 6c 6f 63 6b  feed = feed.lock
3160: 5f 61 72 63 28 29 2e 61 77 61 69 74 3b 0a 09 09  _arc().await;...
3170: 09 09 66 65 65 64 2e 69 6e 73 65 72 74 28 73 6f  ..feed.insert(so
3180: 75 72 63 65 5f 69 64 2c 20 63 68 61 6e 6e 65 6c  urce_id, channel
3190: 29 3b 0a 09 09 09 7d 20 65 6c 73 65 20 7b 0a 09  );....} else {..
31a0: 09 09 09 69 6e 73 65 72 74 65 64 20 3d 20 66 61  ...inserted = fa
31b0: 6c 73 65 3b 0a 09 09 09 7d 0a 09 09 7d 0a 09 09  lse;....}...}...
31c0: 2f 2f 20 69 6e 20 63 61 73 65 20 69 6e 73 65 72  // in case inser
31d0: 74 20 66 61 69 6c 65 64 20 2d 20 77 65 20 6d 69  t failed - we mi
31e0: 73 73 20 74 68 65 20 65 6e 74 72 79 20 77 65 20  ss the entry we 
31f0: 6e 65 65 64 65 64 20 74 6f 20 65 78 70 61 6e 64  needed to expand
3200: 2c 20 72 65 6c 6f 61 64 20 65 76 65 72 79 74 68  , reload everyth
3210: 69 6e 67 20 66 72 6f 6d 0a 09 09 2f 2f 20 64 61  ing from...// da
3220: 74 61 62 61 73 65 0a 09 09 69 66 20 21 69 6e 73  tabase...if !ins
3230: 65 72 74 65 64 20 7b 0a 09 09 09 73 65 6c 66 2e  erted {....self.
3240: 67 65 74 5f 66 65 65 64 73 28 6f 77 6e 65 72 29  get_feeds(owner)
3250: 2e 61 77 61 69 74 2e 73 74 61 63 6b 28 29 3f 3b  .await.stack()?;
3260: 0a 09 09 7d 0a 09 09 4f 6b 28 28 29 29 0a 09 7d  ...}...Ok(())..}
3270: 0a 0a 09 2f 2f 2f 20 52 65 6d 6f 76 65 73 20 66  .../// Removes f
3280: 65 65 64 20 66 72 6f 6d 20 63 61 63 68 65 64 20  eed from cached 
3290: 6c 69 73 74 0a 09 70 75 62 20 61 73 79 6e 63 20  list..pub async 
32a0: 66 6e 20 72 6d 5f 66 65 65 64 20 28 26 73 65 6c  fn rm_feed (&sel
32b0: 66 2c 20 6f 77 6e 65 72 3a 20 69 36 34 2c 20 73  f, owner: i64, s
32c0: 6f 75 72 63 65 5f 69 64 3a 20 26 69 33 32 29 20  ource_id: &i32) 
32d0: 2d 3e 20 52 65 73 75 6c 74 3c 28 29 3e 20 7b 0a  -> Result<()> {.
32e0: 09 09 6c 65 74 20 6d 75 74 20 64 72 6f 70 70 65  ..let mut droppe
32f0: 64 20 3d 20 66 61 6c 73 65 3b 0a 09 09 7b 0a 09  d = false;...{..
3300: 09 09 6c 65 74 20 6d 75 74 20 66 65 65 64 73 20  ..let mut feeds 
3310: 3d 20 73 65 6c 66 2e 66 65 65 64 73 2e 6c 6f 63  = self.feeds.loc
3320: 6b 5f 61 72 63 28 29 2e 61 77 61 69 74 3b 0a 09  k_arc().await;..
3330: 09 09 69 66 20 6c 65 74 20 53 6f 6d 65 28 66 65  ..if let Some(fe
3340: 65 64 29 20 3d 20 66 65 65 64 73 2e 67 65 74 5f  ed) = feeds.get_
3350: 6d 75 74 28 26 6f 77 6e 65 72 29 20 7b 0a 09 09  mut(&owner) {...
3360: 09 09 6c 65 74 20 6d 75 74 20 66 65 65 64 20 3d  ..let mut feed =
3370: 20 66 65 65 64 2e 6c 6f 63 6b 5f 61 72 63 28 29   feed.lock_arc()
3380: 2e 61 77 61 69 74 3b 0a 09 09 09 09 66 65 65 64  .await;.....feed
3390: 2e 72 65 6d 6f 76 65 28 73 6f 75 72 63 65 5f 69  .remove(source_i
33a0: 64 29 3b 0a 09 09 09 09 64 72 6f 70 70 65 64 20  d);.....dropped 
33b0: 3d 20 74 72 75 65 3b 0a 09 09 09 7d 0a 09 09 7d  = true;....}...}
33c0: 0a 09 09 2f 2f 20 69 6e 20 63 61 73 65 20 77 65  ...// in case we
33d0: 20 66 61 69 6c 65 64 20 74 6f 20 66 6f 75 6e 64   failed to found
33e0: 20 66 65 65 64 20 77 65 20 6e 65 65 64 20 74 6f   feed we need to
33f0: 20 72 65 6d 6f 76 65 20 2d 20 6a 75 73 74 20 72   remove - just r
3400: 65 6c 6f 61 64 20 65 76 65 72 79 74 68 69 6e 67  eload everything
3410: 20 66 72 6f 6d 20 64 61 74 61 62 61 73 65 0a 09   from database..
3420: 09 69 66 20 21 64 72 6f 70 70 65 64 20 7b 0a 09  .if !dropped {..
3430: 09 09 73 65 6c 66 2e 67 65 74 5f 66 65 65 64 73  ..self.get_feeds
3440: 28 6f 77 6e 65 72 29 2e 61 77 61 69 74 2e 73 74  (owner).await.st
3450: 61 63 6b 28 29 3f 3b 0a 09 09 7d 0a 09 09 4f 6b  ack()?;...}...Ok
3460: 28 28 29 29 0a 09 7d 0a 7d 0a 0a 69 6d 70 6c 20  (())..}.}..impl 
3470: 55 70 64 61 74 65 48 61 6e 64 6c 65 72 20 66 6f  UpdateHandler fo
3480: 72 20 43 6f 72 65 20 7b 0a 09 2f 2f 2f 20 44 69  r Core {../// Di
3490: 73 70 61 74 63 68 65 73 20 61 6e 20 69 6e 63 6f  spatches an inco
34a0: 6d 69 6e 67 20 54 65 6c 65 67 72 61 6d 20 75 70  ming Telegram up
34b0: 64 61 74 65 20 74 6f 20 61 20 6d 61 74 63 68 69  date to a matchi
34c0: 6e 67 20 63 6f 6d 6d 61 6e 64 20 68 61 6e 64 6c  ng command handl
34d0: 65 72 20 61 6e 64 20 72 65 70 6f 72 74 73 20 68  er and reports h
34e0: 61 6e 64 6c 65 72 20 65 72 72 6f 72 73 20 74 6f  andler errors to
34f0: 20 74 68 65 20 6f 72 69 67 69 6e 61 74 69 6e 67   the originating
3500: 20 63 68 61 74 2e 0a 09 2f 2f 2f 0a 09 2f 2f 2f   chat...///..///
3510: 20 54 68 69 73 20 6d 65 74 68 6f 64 20 69 6e 73   This method ins
3520: 70 65 63 74 73 20 74 68 65 20 75 70 64 61 74 65  pects the update
3530: 3b 20 69 66 20 69 74 20 63 6f 6e 74 61 69 6e 73  ; if it contains
3540: 20 61 20 6d 65 73 73 61 67 65 20 74 68 61 74 20   a message that 
3550: 63 61 6e 20 62 65 20 70 61 72 73 65 64 20 61 73  can be parsed as
3560: 20 61 20 62 6f 74 20 63 6f 6d 6d 61 6e 64 2c 0a   a bot command,.
3570: 09 2f 2f 2f 20 69 74 20 65 78 65 63 75 74 65 73  ./// it executes
3580: 20 74 68 65 20 63 6f 72 72 65 73 70 6f 6e 64 69   the correspondi
3590: 6e 67 20 63 6f 6d 6d 61 6e 64 20 68 61 6e 64 6c  ng command handl
35a0: 65 72 2e 20 49 66 20 74 68 65 20 68 61 6e 64 6c  er. If the handl
35b0: 65 72 20 72 65 74 75 72 6e 73 20 61 6e 20 65 72  er returns an er
35c0: 72 6f 72 2c 20 74 68 65 20 65 72 72 6f 72 20 74  ror, the error t
35d0: 65 78 74 0a 09 2f 2f 2f 20 69 73 20 73 65 6e 74  ext../// is sent
35e0: 20 62 61 63 6b 20 74 6f 20 74 68 65 20 6d 65 73   back to the mes
35f0: 73 61 67 65 27 73 20 63 68 61 74 20 75 73 69 6e  sage's chat usin
3600: 67 20 4d 61 72 6b 64 6f 77 6e 56 32 20 66 6f 72  g MarkdownV2 for
3610: 6d 61 74 74 69 6e 67 2e 20 55 6e 6b 6e 6f 77 6e  matting. Unknown
3620: 20 63 6f 6d 6d 61 6e 64 73 20 70 72 6f 64 75 63   commands produc
3630: 65 20 61 6e 20 65 72 72 6f 72 0a 09 2f 2f 2f 20  e an error../// 
3640: 77 68 69 63 68 20 69 73 20 61 6c 73 6f 20 72 65  which is also re
3650: 70 6f 72 74 65 64 20 74 6f 20 74 68 65 20 63 68  ported to the ch
3660: 61 74 2e 0a 09 61 73 79 6e 63 20 66 6e 20 68 61  at...async fn ha
3670: 6e 64 6c 65 20 28 26 73 65 6c 66 2c 20 75 70 64  ndle (&self, upd
3680: 61 74 65 3a 20 55 70 64 61 74 65 29 20 7b 0a 09  ate: Update) {..
3690: 09 69 66 20 6c 65 74 20 55 70 64 61 74 65 54 79  .if let UpdateTy
36a0: 70 65 3a 3a 4d 65 73 73 61 67 65 28 6d 73 67 29  pe::Message(msg)
36b0: 20 3d 20 75 70 64 61 74 65 2e 75 70 64 61 74 65   = update.update
36c0: 5f 74 79 70 65 20 0a 09 09 09 26 26 20 6c 65 74  _type ....&& let
36d0: 20 4f 6b 28 63 6d 64 29 20 3d 20 43 6f 6d 6d 61   Ok(cmd) = Comma
36e0: 6e 64 3a 3a 74 72 79 5f 66 72 6f 6d 28 2a 6d 73  nd::try_from(*ms
36f0: 67 29 0a 09 09 7b 0a 09 09 09 6c 65 74 20 6d 73  g)...{....let ms
3700: 67 20 3d 20 63 6d 64 2e 67 65 74 5f 6d 65 73 73  g = cmd.get_mess
3710: 61 67 65 28 29 3b 0a 09 09 09 6c 65 74 20 77 6f  age();....let wo
3720: 72 64 73 20 3d 20 63 6d 64 2e 67 65 74 5f 61 72  rds = cmd.get_ar
3730: 67 73 28 29 3b 0a 09 09 09 6c 65 74 20 63 6f 6d  gs();....let com
3740: 6d 61 6e 64 20 3d 20 63 6d 64 2e 67 65 74 5f 6e  mand = cmd.get_n
3750: 61 6d 65 28 29 3b 0a 09 09 09 6c 65 74 20 72 65  ame();....let re
3760: 73 20 3d 20 6d 61 74 63 68 20 63 6f 6d 6d 61 6e  s = match comman
3770: 64 20 7b 0a 09 09 09 09 22 2f 63 68 65 63 6b 22  d {....."/check"
3780: 20 7c 20 22 2f 63 6c 65 61 6e 22 20 7c 20 22 2f   | "/clean" | "/
3790: 65 6e 61 62 6c 65 22 20 7c 20 22 2f 64 65 6c 65  enable" | "/dele
37a0: 74 65 22 20 7c 20 22 2f 64 69 73 61 62 6c 65 22  te" | "/disable"
37b0: 20 3d 3e 20 63 6f 6d 6d 61 6e 64 3a 3a 63 6f 6d   => command::com
37c0: 6d 61 6e 64 28 73 65 6c 66 2c 20 63 6f 6d 6d 61  mand(self, comma
37d0: 6e 64 2c 20 6d 73 67 2c 20 77 6f 72 64 73 29 2e  nd, msg, words).
37e0: 61 77 61 69 74 2c 0a 09 09 09 09 22 2f 73 74 61  await,....."/sta
37f0: 72 74 22 20 3d 3e 20 63 6f 6d 6d 61 6e 64 3a 3a  rt" => command::
3800: 73 74 61 72 74 28 73 65 6c 66 2c 20 6d 73 67 29  start(self, msg)
3810: 2e 61 77 61 69 74 2c 0a 09 09 09 09 22 2f 6c 69  .await,....."/li
3820: 73 74 22 20 3d 3e 20 63 6f 6d 6d 61 6e 64 3a 3a  st" => command::
3830: 6c 69 73 74 28 73 65 6c 66 2c 20 6d 73 67 29 2e  list(self, msg).
3840: 61 77 61 69 74 2c 0a 09 09 09 09 22 2f 74 65 73  await,....."/tes
3850: 74 22 20 3d 3e 20 63 6f 6d 6d 61 6e 64 3a 3a 74  t" => command::t
3860: 65 73 74 28 73 65 6c 66 2c 20 6d 73 67 29 2e 61  est(self, msg).a
3870: 77 61 69 74 2c 0a 09 09 09 09 22 2f 61 64 64 22  wait,....."/add"
3880: 20 7c 20 22 2f 75 70 64 61 74 65 22 20 3d 3e 20   | "/update" => 
3890: 63 6f 6d 6d 61 6e 64 3a 3a 75 70 64 61 74 65 28  command::update(
38a0: 73 65 6c 66 2c 20 63 6f 6d 6d 61 6e 64 2c 20 6d  self, command, m
38b0: 73 67 2c 20 77 6f 72 64 73 29 2e 61 77 61 69 74  sg, words).await
38c0: 2c 0a 09 09 09 09 61 6e 79 20 3d 3e 20 45 72 72  ,.....any => Err
38d0: 28 61 6e 79 68 6f 77 21 28 22 55 6e 6b 6e 6f 77  (anyhow!("Unknow
38e0: 6e 20 63 6f 6d 6d 61 6e 64 3a 20 7b 61 6e 79 7d  n command: {any}
38f0: 22 29 29 2c 0a 09 09 09 7d 3b 0a 09 09 09 69 66  ")),....};....if
3900: 20 6c 65 74 20 45 72 72 28 65 72 72 29 20 3d 20   let Err(err) = 
3910: 72 65 73 20 0a 09 09 09 09 26 26 20 6c 65 74 20  res .....&& let 
3920: 45 72 72 28 65 72 72 32 29 20 3d 20 73 65 6c 66  Err(err2) = self
3930: 2e 74 67 2e 73 65 6e 64 28 4d 79 4d 65 73 73 61  .tg.send(MyMessa
3940: 67 65 3a 3a 74 65 78 74 5f 74 6f 28 0a 09 09 09  ge::text_to(....
3950: 09 09 66 6f 72 6d 61 74 21 28 22 5c 5c 23 65 72  ..format!("\\#er
3960: 72 6f 72 5c 6e 60 60 60 5c 6e 7b 65 72 72 7d 5c  ror\n```\n{err}\
3970: 6e 60 60 60 22 29 2c 0a 09 09 09 09 09 6d 73 67  n```"),......msg
3980: 2e 63 68 61 74 2e 67 65 74 5f 69 64 28 29 2c 0a  .chat.get_id(),.
3990: 09 09 09 09 29 29 2e 61 77 61 69 74 0a 09 09 09  ....)).await....
39a0: 7b 0a 09 09 09 09 64 62 67 21 28 65 72 72 32 29  {.....dbg!(err2)
39b0: 3b 0a 09 09 09 7d 0a 09 09 7d 20 2f 2f 20 54 4f  ;....}...} // TO
39c0: 44 4f 3a 20 64 65 62 75 67 20 6c 6f 67 20 66 6f  DO: debug log fo
39d0: 72 20 73 6b 69 70 70 65 64 20 75 70 64 61 74 65  r skipped update
39e0: 73 3f 3b 0a 09 7d 0a 7d 0a                       s?;..}.}.