Hex Artifact Content
Logged in as anonymous

Artifact 07fc2cd5fdf8bd8a582edb844025b8dedbbb97446bc488f49f5c7f48e34dd14c:


0000: 75 73 65 20 61 6e 79 68 6f 77 3a 3a 7b 61 6e 79  use anyhow::{any
0010: 68 6f 77 2c 20 62 61 69 6c 2c 20 43 6f 6e 74 65  how, bail, Conte
0020: 78 74 2c 20 52 65 73 75 6c 74 7d 3b 0a 75 73 65  xt, Result};.use
0030: 20 61 73 79 6e 63 5f 73 74 64 3a 3a 74 61 73 6b   async_std::task
0040: 3b 0a 75 73 65 20 63 68 72 6f 6e 6f 3a 3a 44 61  ;.use chrono::Da
0050: 74 65 54 69 6d 65 3b 0a 75 73 65 20 73 71 6c 78  teTime;.use sqlx
0060: 3a 3a 70 6f 73 74 67 72 65 73 3a 3a 50 67 50 6f  ::postgres::PgPo
0070: 6f 6c 4f 70 74 69 6f 6e 73 3b 0a 75 73 65 20 74  olOptions;.use t
0080: 65 6c 65 67 72 61 6d 5f 62 6f 74 3a 3a 7b 0a 09  elegram_bot::{..
0090: 5f 62 61 73 65 3a 3a 45 72 72 6f 72 20 61 73 20  _base::Error as 
00a0: 54 67 72 45 72 72 6f 72 2c 0a 09 45 72 72 6f 72  TgrError,..Error
00b0: 20 61 73 20 54 67 45 72 72 6f 72 2c 0a 7d 3b 0a   as TgError,.};.
00c0: 75 73 65 20 74 68 69 73 65 72 72 6f 72 3a 3a 45  use thiserror::E
00d0: 72 72 6f 72 3b 0a 0a 75 73 65 20 73 74 64 3a 3a  rror;..use std::
00e0: 7b 0a 09 62 6f 72 72 6f 77 3a 3a 43 6f 77 2c 0a  {..borrow::Cow,.
00f0: 09 63 6f 6c 6c 65 63 74 69 6f 6e 73 3a 3a 7b 0a  .collections::{.
0100: 09 09 42 54 72 65 65 4d 61 70 2c 0a 09 09 48 61  ..BTreeMap,...Ha
0110: 73 68 53 65 74 2c 0a 09 7d 2c 0a 09 6e 75 6d 3a  shSet,..},..num:
0120: 3a 54 72 79 46 72 6f 6d 49 6e 74 45 72 72 6f 72  :TryFromIntError
0130: 2c 0a 09 73 79 6e 63 3a 3a 7b 0a 09 09 41 72 63  ,..sync::{...Arc
0140: 2c 0a 09 09 4d 75 74 65 78 0a 09 7d 2c 0a 7d 3b  ,...Mutex..},.};
0150: 0a 0a 23 5b 64 65 72 69 76 65 28 45 72 72 6f 72  ..#[derive(Error
0160: 2c 20 44 65 62 75 67 29 5d 0a 70 75 62 20 65 6e  , Debug)].pub en
0170: 75 6d 20 52 73 73 45 72 72 6f 72 20 7b 0a 09 23  um RssError {..#
0180: 5b 65 72 72 6f 72 28 74 72 61 6e 73 70 61 72 65  [error(transpare
0190: 6e 74 29 5d 0a 09 54 67 28 23 5b 66 72 6f 6d 5d  nt)]..Tg(#[from]
01a0: 20 54 67 45 72 72 6f 72 29 2c 0a 09 23 5b 65 72   TgError),..#[er
01b0: 72 6f 72 28 74 72 61 6e 73 70 61 72 65 6e 74 29  ror(transparent)
01c0: 5d 0a 09 49 6e 74 28 23 5b 66 72 6f 6d 5d 20 54  ]..Int(#[from] T
01d0: 72 79 46 72 6f 6d 49 6e 74 45 72 72 6f 72 29 2c  ryFromIntError),
01e0: 0a 7d 0a 0a 23 5b 64 65 72 69 76 65 28 43 6c 6f  .}..#[derive(Clo
01f0: 6e 65 29 5d 0a 70 75 62 20 73 74 72 75 63 74 20  ne)].pub struct 
0200: 43 6f 72 65 20 7b 0a 09 6f 77 6e 65 72 5f 63 68  Core {..owner_ch
0210: 61 74 3a 20 74 65 6c 65 67 72 61 6d 5f 62 6f 74  at: telegram_bot
0220: 3a 3a 55 73 65 72 49 64 2c 0a 09 70 75 62 20 74  ::UserId,..pub t
0230: 67 3a 20 74 65 6c 65 67 72 61 6d 5f 62 6f 74 3a  g: telegram_bot:
0240: 3a 41 70 69 2c 0a 09 70 75 62 20 6d 79 3a 20 74  :Api,..pub my: t
0250: 65 6c 65 67 72 61 6d 5f 62 6f 74 3a 3a 55 73 65  elegram_bot::Use
0260: 72 2c 0a 09 70 6f 6f 6c 3a 20 73 71 6c 78 3a 3a  r,..pool: sqlx::
0270: 50 6f 6f 6c 3c 73 71 6c 78 3a 3a 50 6f 73 74 67  Pool<sqlx::Postg
0280: 72 65 73 3e 2c 0a 09 73 6f 75 72 63 65 73 3a 20  res>,..sources: 
0290: 41 72 63 3c 4d 75 74 65 78 3c 48 61 73 68 53 65  Arc<Mutex<HashSe
02a0: 74 3c 41 72 63 3c 69 33 32 3e 3e 3e 3e 2c 0a 09  t<Arc<i32>>>>,..
02b0: 68 74 74 70 5f 63 6c 69 65 6e 74 3a 20 72 65 71  http_client: req
02c0: 77 65 73 74 3a 3a 43 6c 69 65 6e 74 2c 0a 7d 0a  west::Client,.}.
02d0: 0a 69 6d 70 6c 20 43 6f 72 65 20 7b 0a 09 70 75  .impl Core {..pu
02e0: 62 20 66 6e 20 6e 65 77 28 73 65 74 74 69 6e 67  b fn new(setting
02f0: 73 3a 20 63 6f 6e 66 69 67 3a 3a 43 6f 6e 66 69  s: config::Confi
0300: 67 29 20 2d 3e 20 52 65 73 75 6c 74 3c 41 72 63  g) -> Result<Arc
0310: 3c 43 6f 72 65 3e 3e 20 7b 0a 09 09 6c 65 74 20  <Core>> {...let 
0320: 6f 77 6e 65 72 20 3d 20 73 65 74 74 69 6e 67 73  owner = settings
0330: 2e 67 65 74 5f 69 6e 74 28 22 6f 77 6e 65 72 22  .get_int("owner"
0340: 29 3f 3b 0a 09 09 6c 65 74 20 61 70 69 5f 6b 65  )?;...let api_ke
0350: 79 20 3d 20 73 65 74 74 69 6e 67 73 2e 67 65 74  y = settings.get
0360: 5f 73 74 72 69 6e 67 28 22 61 70 69 5f 6b 65 79  _string("api_key
0370: 22 29 3f 3b 0a 09 09 6c 65 74 20 74 67 20 3d 20  ")?;...let tg = 
0380: 74 65 6c 65 67 72 61 6d 5f 62 6f 74 3a 3a 41 70  telegram_bot::Ap
0390: 69 3a 3a 6e 65 77 28 61 70 69 5f 6b 65 79 29 3b  i::new(api_key);
03a0: 0a 09 09 6c 65 74 20 74 67 5f 63 6c 6f 6e 65 64  ...let tg_cloned
03b0: 20 3d 20 74 67 2e 63 6c 6f 6e 65 28 29 3b 0a 0a   = tg.clone();..
03c0: 09 09 6c 65 74 20 6d 75 74 20 63 6c 69 65 6e 74  ..let mut client
03d0: 20 3d 20 72 65 71 77 65 73 74 3a 3a 43 6c 69 65   = reqwest::Clie
03e0: 6e 74 3a 3a 62 75 69 6c 64 65 72 28 29 3b 0a 09  nt::builder();..
03f0: 09 69 66 20 6c 65 74 20 4f 6b 28 70 72 6f 78 79  .if let Ok(proxy
0400: 29 20 3d 20 73 65 74 74 69 6e 67 73 2e 67 65 74  ) = settings.get
0410: 5f 73 74 72 69 6e 67 28 22 70 72 6f 78 79 22 29  _string("proxy")
0420: 20 7b 0a 09 09 09 6c 65 74 20 70 72 6f 78 79 20   {....let proxy 
0430: 3d 20 72 65 71 77 65 73 74 3a 3a 50 72 6f 78 79  = reqwest::Proxy
0440: 3a 3a 61 6c 6c 28 70 72 6f 78 79 29 3f 3b 0a 09  ::all(proxy)?;..
0450: 09 09 63 6c 69 65 6e 74 20 3d 20 63 6c 69 65 6e  ..client = clien
0460: 74 2e 70 72 6f 78 79 28 70 72 6f 78 79 29 3b 0a  t.proxy(proxy);.
0470: 09 09 7d 0a 09 09 6c 65 74 20 68 74 74 70 5f 63  ..}...let http_c
0480: 6c 69 65 6e 74 20 3d 20 63 6c 69 65 6e 74 2e 62  lient = client.b
0490: 75 69 6c 64 28 29 3f 3b 0a 09 09 6c 65 74 20 63  uild()?;...let c
04a0: 6f 72 65 20 3d 20 41 72 63 3a 3a 6e 65 77 28 43  ore = Arc::new(C
04b0: 6f 72 65 20 7b 0a 09 09 09 74 67 2c 0a 09 09 09  ore {....tg,....
04c0: 6d 79 3a 20 74 61 73 6b 3a 3a 62 6c 6f 63 6b 5f  my: task::block_
04d0: 6f 6e 28 61 73 79 6e 63 20 7b 0a 09 09 09 09 74  on(async {.....t
04e0: 67 5f 63 6c 6f 6e 65 64 2e 73 65 6e 64 28 74 65  g_cloned.send(te
04f0: 6c 65 67 72 61 6d 5f 62 6f 74 3a 3a 47 65 74 4d  legram_bot::GetM
0500: 65 29 2e 61 77 61 69 74 0a 09 09 09 7d 29 3f 2c  e).await....})?,
0510: 0a 09 09 09 6f 77 6e 65 72 5f 63 68 61 74 3a 20  ....owner_chat: 
0520: 74 65 6c 65 67 72 61 6d 5f 62 6f 74 3a 3a 55 73  telegram_bot::Us
0530: 65 72 49 64 3a 3a 6e 65 77 28 6f 77 6e 65 72 29  erId::new(owner)
0540: 2c 0a 09 09 09 70 6f 6f 6c 3a 20 50 67 50 6f 6f  ,....pool: PgPoo
0550: 6c 4f 70 74 69 6f 6e 73 3a 3a 6e 65 77 28 29 0a  lOptions::new().
0560: 09 09 09 09 2e 6d 61 78 5f 63 6f 6e 6e 65 63 74  .....max_connect
0570: 69 6f 6e 73 28 35 29 0a 09 09 09 09 2e 61 63 71  ions(5)......acq
0580: 75 69 72 65 5f 74 69 6d 65 6f 75 74 28 73 74 64  uire_timeout(std
0590: 3a 3a 74 69 6d 65 3a 3a 44 75 72 61 74 69 6f 6e  ::time::Duration
05a0: 3a 3a 6e 65 77 28 33 30 30 2c 20 30 29 29 0a 09  ::new(300, 0))..
05b0: 09 09 09 2e 69 64 6c 65 5f 74 69 6d 65 6f 75 74  ....idle_timeout
05c0: 28 73 74 64 3a 3a 74 69 6d 65 3a 3a 44 75 72 61  (std::time::Dura
05d0: 74 69 6f 6e 3a 3a 6e 65 77 28 36 30 2c 20 30 29  tion::new(60, 0)
05e0: 29 0a 09 09 09 09 2e 63 6f 6e 6e 65 63 74 5f 6c  )......connect_l
05f0: 61 7a 79 28 26 73 65 74 74 69 6e 67 73 2e 67 65  azy(&settings.ge
0600: 74 5f 73 74 72 69 6e 67 28 22 70 67 22 29 3f 29  t_string("pg")?)
0610: 3f 2c 0a 09 09 09 73 6f 75 72 63 65 73 3a 20 41  ?,....sources: A
0620: 72 63 3a 3a 6e 65 77 28 4d 75 74 65 78 3a 3a 6e  rc::new(Mutex::n
0630: 65 77 28 48 61 73 68 53 65 74 3a 3a 6e 65 77 28  ew(HashSet::new(
0640: 29 29 29 2c 0a 09 09 09 68 74 74 70 5f 63 6c 69  ))),....http_cli
0650: 65 6e 74 2c 0a 09 09 7d 29 3b 0a 09 09 6c 65 74  ent,...});...let
0660: 20 63 6c 6f 6e 65 20 3d 20 63 6f 72 65 2e 63 6c   clone = core.cl
0670: 6f 6e 65 28 29 3b 0a 09 09 74 61 73 6b 3a 3a 73  one();...task::s
0680: 70 61 77 6e 28 61 73 79 6e 63 20 6d 6f 76 65 20  pawn(async move 
0690: 7b 0a 09 09 09 6c 6f 6f 70 20 7b 0a 09 09 09 09  {....loop {.....
06a0: 6c 65 74 20 64 65 6c 61 79 20 3d 20 6d 61 74 63  let delay = matc
06b0: 68 20 26 63 6c 6f 6e 65 2e 61 75 74 6f 66 65 74  h &clone.autofet
06c0: 63 68 28 29 2e 61 77 61 69 74 20 7b 0a 09 09 09  ch().await {....
06d0: 09 09 45 72 72 28 65 72 72 29 20 3d 3e 20 7b 0a  ..Err(err) => {.
06e0: 09 09 09 09 09 09 69 66 20 6c 65 74 20 45 72 72  ......if let Err
06f0: 28 65 72 72 29 20 3d 20 63 6c 6f 6e 65 2e 73 65  (err) = clone.se
0700: 6e 64 28 66 6f 72 6d 61 74 21 28 22 f0 9f 9b 91  nd(format!("šŸ›‘
0710: 20 7b 3a 3f 7d 22 2c 20 65 72 72 29 2c 20 4e 6f   {:?}", err), No
0720: 6e 65 2c 20 4e 6f 6e 65 29 2e 61 77 61 69 74 20  ne, None).await 
0730: 7b 0a 09 09 09 09 09 09 09 65 70 72 69 6e 74 6c  {........eprintl
0740: 6e 21 28 22 41 75 74 6f 66 65 74 63 68 20 65 72  n!("Autofetch er
0750: 72 6f 72 3a 20 7b 7d 22 2c 20 65 72 72 29 3b 0a  ror: {}", err);.
0760: 09 09 09 09 09 09 7d 3b 0a 09 09 09 09 09 09 73  ......};.......s
0770: 74 64 3a 3a 74 69 6d 65 3a 3a 44 75 72 61 74 69  td::time::Durati
0780: 6f 6e 3a 3a 66 72 6f 6d 5f 73 65 63 73 28 36 30  on::from_secs(60
0790: 29 0a 09 09 09 09 09 7d 2c 0a 09 09 09 09 09 4f  )......},......O
07a0: 6b 28 74 69 6d 65 29 20 3d 3e 20 2a 74 69 6d 65  k(time) => *time
07b0: 2c 0a 09 09 09 09 7d 3b 0a 09 09 09 09 74 61 73  ,.....};.....tas
07c0: 6b 3a 3a 73 6c 65 65 70 28 64 65 6c 61 79 29 2e  k::sleep(delay).
07d0: 61 77 61 69 74 3b 0a 09 09 09 7d 0a 09 09 7d 29  await;....}...})
07e0: 3b 0a 09 09 4f 6b 28 63 6f 72 65 29 0a 09 7d 0a  ;...Ok(core)..}.
07f0: 0a 09 70 75 62 20 66 6e 20 73 74 72 65 61 6d 28  ..pub fn stream(
0800: 26 73 65 6c 66 29 20 2d 3e 20 74 65 6c 65 67 72  &self) -> telegr
0810: 61 6d 5f 62 6f 74 3a 3a 55 70 64 61 74 65 73 53  am_bot::UpdatesS
0820: 74 72 65 61 6d 20 7b 0a 09 09 73 65 6c 66 2e 74  tream {...self.t
0830: 67 2e 73 74 72 65 61 6d 28 29 0a 09 7d 0a 0a 09  g.stream()..}...
0840: 70 75 62 20 61 73 79 6e 63 20 66 6e 20 73 65 6e  pub async fn sen
0850: 64 3c 27 61 2c 20 53 3e 28 26 73 65 6c 66 2c 20  d<'a, S>(&self, 
0860: 6d 73 67 3a 20 53 2c 20 74 61 72 67 65 74 3a 20  msg: S, target: 
0870: 4f 70 74 69 6f 6e 3c 74 65 6c 65 67 72 61 6d 5f  Option<telegram_
0880: 62 6f 74 3a 3a 55 73 65 72 49 64 3e 2c 20 6d 6f  bot::UserId>, mo
0890: 64 65 3a 20 4f 70 74 69 6f 6e 3c 74 65 6c 65 67  de: Option<teleg
08a0: 72 61 6d 5f 62 6f 74 3a 3a 74 79 70 65 73 3a 3a  ram_bot::types::
08b0: 50 61 72 73 65 4d 6f 64 65 3e 29 20 2d 3e 20 52  ParseMode>) -> R
08c0: 65 73 75 6c 74 3c 28 29 3e 0a 09 77 68 65 72 65  esult<()>..where
08d0: 20 53 3a 20 49 6e 74 6f 3c 43 6f 77 3c 27 61 2c   S: Into<Cow<'a,
08e0: 20 73 74 72 3e 3e 20 7b 0a 09 09 6c 65 74 20 6d   str>> {...let m
08f0: 6f 64 65 20 3d 20 6d 6f 64 65 2e 75 6e 77 72 61  ode = mode.unwra
0900: 70 5f 6f 72 28 74 65 6c 65 67 72 61 6d 5f 62 6f  p_or(telegram_bo
0910: 74 3a 3a 74 79 70 65 73 3a 3a 50 61 72 73 65 4d  t::types::ParseM
0920: 6f 64 65 3a 3a 48 74 6d 6c 29 3b 0a 09 09 6c 65  ode::Html);...le
0930: 74 20 74 61 72 67 65 74 20 3d 20 74 61 72 67 65  t target = targe
0940: 74 2e 75 6e 77 72 61 70 5f 6f 72 28 73 65 6c 66  t.unwrap_or(self
0950: 2e 6f 77 6e 65 72 5f 63 68 61 74 29 3b 0a 09 09  .owner_chat);...
0960: 73 65 6c 66 2e 72 65 71 75 65 73 74 28 74 65 6c  self.request(tel
0970: 65 67 72 61 6d 5f 62 6f 74 3a 3a 53 65 6e 64 4d  egram_bot::SendM
0980: 65 73 73 61 67 65 3a 3a 6e 65 77 28 74 61 72 67  essage::new(targ
0990: 65 74 2c 20 6d 73 67 29 2e 70 61 72 73 65 5f 6d  et, msg).parse_m
09a0: 6f 64 65 28 6d 6f 64 65 29 29 2e 61 77 61 69 74  ode(mode)).await
09b0: 3f 3b 0a 09 09 4f 6b 28 28 29 29 0a 09 7d 0a 0a  ?;...Ok(())..}..
09c0: 09 70 75 62 20 61 73 79 6e 63 20 66 6e 20 72 65  .pub async fn re
09d0: 71 75 65 73 74 3c 52 65 71 3a 20 74 65 6c 65 67  quest<Req: teleg
09e0: 72 61 6d 5f 62 6f 74 3a 3a 52 65 71 75 65 73 74  ram_bot::Request
09f0: 3e 20 28 26 73 65 6c 66 2c 20 72 65 71 3a 20 52  > (&self, req: R
0a00: 65 71 29 20 2d 3e 20 52 65 73 75 6c 74 3c 3c 52  eq) -> Result<<R
0a10: 65 71 3a 3a 52 65 73 70 6f 6e 73 65 20 61 73 20  eq::Response as 
0a20: 74 65 6c 65 67 72 61 6d 5f 62 6f 74 3a 3a 52 65  telegram_bot::Re
0a30: 73 70 6f 6e 73 65 54 79 70 65 3e 3a 3a 54 79 70  sponseType>::Typ
0a40: 65 2c 20 52 73 73 45 72 72 6f 72 3e 20 7b 0a 09  e, RssError> {..
0a50: 09 6c 6f 6f 70 20 7b 0a 09 09 09 6c 65 74 20 72  .loop {....let r
0a60: 65 73 20 3d 20 73 65 6c 66 2e 74 67 2e 73 65 6e  es = self.tg.sen
0a70: 64 28 26 72 65 71 29 2e 61 77 61 69 74 3b 0a 09  d(&req).await;..
0a80: 09 09 6d 61 74 63 68 20 72 65 73 20 7b 0a 09 09  ..match res {...
0a90: 09 09 4f 6b 28 5f 29 20 3d 3e 20 72 65 74 75 72  ..Ok(_) => retur
0aa0: 6e 20 4f 6b 28 72 65 73 3f 29 2c 0a 09 09 09 09  n Ok(res?),.....
0ab0: 45 72 72 28 65 72 72 29 20 3d 3e 20 7b 0a 09 09  Err(err) => {...
0ac0: 09 09 09 6d 61 74 63 68 20 26 65 72 72 20 7b 0a  ...match &err {.
0ad0: 09 09 09 09 09 09 54 67 45 72 72 6f 72 3a 3a 52  ......TgError::R
0ae0: 61 77 28 54 67 72 45 72 72 6f 72 3a 3a 54 65 6c  aw(TgrError::Tel
0af0: 65 67 72 61 6d 45 72 72 6f 72 20 7b 20 64 65 73  egramError { des
0b00: 63 72 69 70 74 69 6f 6e 3a 20 5f 2c 20 70 61 72  cription: _, par
0b10: 61 6d 65 74 65 72 73 3a 20 53 6f 6d 65 28 70 61  ameters: Some(pa
0b20: 72 61 6d 73 29 20 7d 29 20 3d 3e 20 7b 0a 09 09  rams) }) => {...
0b30: 09 09 09 09 09 69 66 20 6c 65 74 20 53 6f 6d 65  .....if let Some
0b40: 28 64 65 6c 61 79 29 20 3d 20 70 61 72 61 6d 73  (delay) = params
0b50: 2e 72 65 74 72 79 5f 61 66 74 65 72 20 7b 0a 09  .retry_after {..
0b60: 09 09 09 09 09 09 09 70 72 69 6e 74 6c 6e 21 28  .......println!(
0b70: 22 54 68 72 6f 74 74 6c 65 64 2c 20 77 61 69 74  "Throttled, wait
0b80: 69 6e 67 20 7b 7d 20 73 65 6e 63 6f 6e 64 73 2e  ing {} senconds.
0b90: 22 2c 20 64 65 6c 61 79 29 3b 0a 09 09 09 09 09  ", delay);......
0ba0: 09 09 09 74 61 73 6b 3a 3a 73 6c 65 65 70 28 73  ...task::sleep(s
0bb0: 74 64 3a 3a 74 69 6d 65 3a 3a 44 75 72 61 74 69  td::time::Durati
0bc0: 6f 6e 3a 3a 66 72 6f 6d 5f 73 65 63 73 28 64 65  on::from_secs(de
0bd0: 6c 61 79 2e 74 72 79 5f 69 6e 74 6f 28 29 3f 29  lay.try_into()?)
0be0: 29 2e 61 77 61 69 74 3b 0a 09 09 09 09 09 09 09  ).await;........
0bf0: 7d 20 65 6c 73 65 20 7b 0a 09 09 09 09 09 09 09  } else {........
0c00: 09 72 65 74 75 72 6e 20 45 72 72 28 65 72 72 2e  .return Err(err.
0c10: 69 6e 74 6f 28 29 29 3b 0a 09 09 09 09 09 09 09  into());........
0c20: 7d 0a 09 09 09 09 09 09 7d 2c 0a 09 09 09 09 09  }.......},......
0c30: 09 5f 20 3d 3e 20 72 65 74 75 72 6e 20 45 72 72  ._ => return Err
0c40: 28 65 72 72 2e 69 6e 74 6f 28 29 29 2c 0a 09 09  (err.into()),...
0c50: 09 09 09 7d 0a 09 09 09 09 7d 2c 0a 09 09 09 7d  ...}.....},....}
0c60: 3b 0a 09 09 7d 0a 09 7d 0a 0a 09 70 75 62 20 61  ;...}..}...pub a
0c70: 73 79 6e 63 20 66 6e 20 63 68 65 63 6b 3c 53 3e  sync fn check<S>
0c80: 28 26 73 65 6c 66 2c 20 69 64 3a 20 26 69 33 32  (&self, id: &i32
0c90: 2c 20 6f 77 6e 65 72 3a 20 53 2c 20 72 65 61 6c  , owner: S, real
0ca0: 3a 20 62 6f 6f 6c 29 20 2d 3e 20 52 65 73 75 6c  : bool) -> Resul
0cb0: 74 3c 43 6f 77 3c 27 5f 2c 20 73 74 72 3e 3e 0a  t<Cow<'_, str>>.
0cc0: 09 77 68 65 72 65 20 53 3a 20 49 6e 74 6f 3c 69  .where S: Into<i
0cd0: 36 34 3e 20 7b 0a 09 09 6c 65 74 20 6f 77 6e 65  64> {...let owne
0ce0: 72 20 3d 20 6f 77 6e 65 72 2e 69 6e 74 6f 28 29  r = owner.into()
0cf0: 3b 0a 09 09 6c 65 74 20 6d 75 74 20 70 6f 73 74  ;...let mut post
0d00: 65 64 3a 20 69 33 32 20 3d 20 30 3b 0a 09 09 6c  ed: i32 = 0;...l
0d10: 65 74 20 6d 75 74 20 63 6f 6e 6e 20 3d 20 73 65  et mut conn = se
0d20: 6c 66 2e 70 6f 6f 6c 2e 61 63 71 75 69 72 65 28  lf.pool.acquire(
0d30: 29 2e 61 77 61 69 74 3f 3b 0a 0a 09 09 6c 65 74  ).await?;....let
0d40: 20 69 64 20 3d 20 7b 0a 09 09 09 6c 65 74 20 6d   id = {....let m
0d50: 75 74 20 73 65 74 20 3d 20 73 65 6c 66 2e 73 6f  ut set = self.so
0d60: 75 72 63 65 73 2e 6c 6f 63 6b 28 29 2e 75 6e 77  urces.lock().unw
0d70: 72 61 70 28 29 3b 0a 09 09 09 6d 61 74 63 68 20  rap();....match 
0d80: 73 65 74 2e 67 65 74 28 69 64 29 20 7b 0a 09 09  set.get(id) {...
0d90: 09 09 53 6f 6d 65 28 69 64 29 20 3d 3e 20 69 64  ..Some(id) => id
0da0: 2e 63 6c 6f 6e 65 28 29 2c 0a 09 09 09 09 4e 6f  .clone(),.....No
0db0: 6e 65 20 3d 3e 20 7b 0a 09 09 09 09 09 6c 65 74  ne => {......let
0dc0: 20 69 64 20 3d 20 41 72 63 3a 3a 6e 65 77 28 2a   id = Arc::new(*
0dd0: 69 64 29 3b 0a 09 09 09 09 09 73 65 74 2e 69 6e  id);......set.in
0de0: 73 65 72 74 28 69 64 2e 63 6c 6f 6e 65 28 29 29  sert(id.clone())
0df0: 3b 0a 09 09 09 09 09 69 64 2e 63 6c 6f 6e 65 28  ;......id.clone(
0e00: 29 0a 09 09 09 09 7d 2c 0a 09 09 09 7d 0a 09 09  ).....},....}...
0e10: 7d 3b 0a 09 09 6c 65 74 20 63 6f 75 6e 74 20 3d  };...let count =
0e20: 20 41 72 63 3a 3a 73 74 72 6f 6e 67 5f 63 6f 75   Arc::strong_cou
0e30: 6e 74 28 26 69 64 29 3b 0a 09 09 69 66 20 63 6f  nt(&id);...if co
0e40: 75 6e 74 20 3d 3d 20 32 20 7b 0a 09 09 09 6c 65  unt == 2 {....le
0e50: 74 20 73 6f 75 72 63 65 20 3d 20 73 71 6c 78 3a  t source = sqlx:
0e60: 3a 71 75 65 72 79 21 28 22 73 65 6c 65 63 74 20  :query!("select 
0e70: 73 6f 75 72 63 65 5f 69 64 2c 20 63 68 61 6e 6e  source_id, chann
0e80: 65 6c 5f 69 64 2c 20 75 72 6c 2c 20 69 76 5f 68  el_id, url, iv_h
0e90: 61 73 68 2c 20 6f 77 6e 65 72 2c 20 75 72 6c 5f  ash, owner, url_
0ea0: 72 65 20 66 72 6f 6d 20 72 73 73 74 67 5f 73 6f  re from rsstg_so
0eb0: 75 72 63 65 20 77 68 65 72 65 20 73 6f 75 72 63  urce where sourc
0ec0: 65 5f 69 64 20 3d 20 24 31 20 61 6e 64 20 6f 77  e_id = $1 and ow
0ed0: 6e 65 72 20 3d 20 24 32 22 2c 0a 09 09 09 09 2a  ner = $2",.....*
0ee0: 69 64 2c 20 6f 77 6e 65 72 29 2e 66 65 74 63 68  id, owner).fetch
0ef0: 5f 6f 6e 65 28 26 6d 75 74 20 2a 63 6f 6e 6e 29  _one(&mut *conn)
0f00: 2e 61 77 61 69 74 3f 3b 0a 09 09 09 6c 65 74 20  .await?;....let 
0f10: 64 65 73 74 69 6e 61 74 69 6f 6e 20 3d 20 6d 61  destination = ma
0f20: 74 63 68 20 72 65 61 6c 20 7b 0a 09 09 09 09 74  tch real {.....t
0f30: 72 75 65 20 3d 3e 20 74 65 6c 65 67 72 61 6d 5f  rue => telegram_
0f40: 62 6f 74 3a 3a 55 73 65 72 49 64 3a 3a 6e 65 77  bot::UserId::new
0f50: 28 73 6f 75 72 63 65 2e 63 68 61 6e 6e 65 6c 5f  (source.channel_
0f60: 69 64 29 2c 0a 09 09 09 09 66 61 6c 73 65 20 3d  id),.....false =
0f70: 3e 20 74 65 6c 65 67 72 61 6d 5f 62 6f 74 3a 3a  > telegram_bot::
0f80: 55 73 65 72 49 64 3a 3a 6e 65 77 28 73 6f 75 72  UserId::new(sour
0f90: 63 65 2e 6f 77 6e 65 72 29 2c 0a 09 09 09 7d 3b  ce.owner),....};
0fa0: 0a 09 09 09 6c 65 74 20 6d 75 74 20 74 68 69 73  ....let mut this
0fb0: 5f 66 65 74 63 68 3a 20 4f 70 74 69 6f 6e 3c 44  _fetch: Option<D
0fc0: 61 74 65 54 69 6d 65 3c 63 68 72 6f 6e 6f 3a 3a  ateTime<chrono::
0fd0: 46 69 78 65 64 4f 66 66 73 65 74 3e 3e 20 3d 20  FixedOffset>> = 
0fe0: 4e 6f 6e 65 3b 0a 09 09 09 6c 65 74 20 6d 75 74  None;....let mut
0ff0: 20 70 6f 73 74 73 3a 20 42 54 72 65 65 4d 61 70   posts: BTreeMap
1000: 3c 44 61 74 65 54 69 6d 65 3c 63 68 72 6f 6e 6f  <DateTime<chrono
1010: 3a 3a 46 69 78 65 64 4f 66 66 73 65 74 3e 2c 20  ::FixedOffset>, 
1020: 53 74 72 69 6e 67 3e 20 3d 20 42 54 72 65 65 4d  String> = BTreeM
1030: 61 70 3a 3a 6e 65 77 28 29 3b 0a 0a 09 09 09 6c  ap::new();.....l
1040: 65 74 20 72 65 73 70 6f 6e 73 65 20 3d 20 73 65  et response = se
1050: 6c 66 2e 68 74 74 70 5f 63 6c 69 65 6e 74 2e 67  lf.http_client.g
1060: 65 74 28 26 73 6f 75 72 63 65 2e 75 72 6c 29 2e  et(&source.url).
1070: 73 65 6e 64 28 29 2e 61 77 61 69 74 3f 3b 0a 09  send().await?;..
1080: 09 09 6c 65 74 20 73 74 61 74 75 73 20 3d 20 72  ..let status = r
1090: 65 73 70 6f 6e 73 65 2e 73 74 61 74 75 73 28 29  esponse.status()
10a0: 3b 0a 09 09 09 6c 65 74 20 63 6f 6e 74 65 6e 74  ;....let content
10b0: 20 3d 20 72 65 73 70 6f 6e 73 65 2e 62 79 74 65   = response.byte
10c0: 73 28 29 2e 61 77 61 69 74 3f 3b 0a 09 09 09 6d  s().await?;....m
10d0: 61 74 63 68 20 72 73 73 3a 3a 43 68 61 6e 6e 65  atch rss::Channe
10e0: 6c 3a 3a 72 65 61 64 5f 66 72 6f 6d 28 26 63 6f  l::read_from(&co
10f0: 6e 74 65 6e 74 5b 2e 2e 5d 29 20 7b 0a 09 09 09  ntent[..]) {....
1100: 09 4f 6b 28 66 65 65 64 29 20 3d 3e 20 7b 0a 09  .Ok(feed) => {..
1110: 09 09 09 09 66 6f 72 20 69 74 65 6d 20 69 6e 20  ....for item in 
1120: 66 65 65 64 2e 69 74 65 6d 73 28 29 20 7b 0a 09  feed.items() {..
1130: 09 09 09 09 09 69 66 20 6c 65 74 20 53 6f 6d 65  .....if let Some
1140: 28 6c 69 6e 6b 29 20 3d 20 69 74 65 6d 2e 6c 69  (link) = item.li
1150: 6e 6b 28 29 20 7b 0a 09 09 09 09 09 09 09 6c 65  nk() {........le
1160: 74 20 64 61 74 65 20 3d 20 6d 61 74 63 68 20 69  t date = match i
1170: 74 65 6d 2e 70 75 62 5f 64 61 74 65 28 29 20 7b  tem.pub_date() {
1180: 0a 09 09 09 09 09 09 09 09 53 6f 6d 65 28 66 65  .........Some(fe
1190: 65 64 5f 64 61 74 65 29 20 3d 3e 20 44 61 74 65  ed_date) => Date
11a0: 54 69 6d 65 3a 3a 70 61 72 73 65 5f 66 72 6f 6d  Time::parse_from
11b0: 5f 72 66 63 32 38 32 32 28 66 65 65 64 5f 64 61  _rfc2822(feed_da
11c0: 74 65 29 2c 0a 09 09 09 09 09 09 09 09 4e 6f 6e  te),.........Non
11d0: 65 20 3d 3e 20 44 61 74 65 54 69 6d 65 3a 3a 70  e => DateTime::p
11e0: 61 72 73 65 5f 66 72 6f 6d 5f 72 66 63 33 33 33  arse_from_rfc333
11f0: 39 28 26 69 74 65 6d 2e 64 75 62 6c 69 6e 5f 63  9(&item.dublin_c
1200: 6f 72 65 5f 65 78 74 28 29 2e 75 6e 77 72 61 70  ore_ext().unwrap
1210: 28 29 2e 64 61 74 65 73 28 29 5b 30 5d 29 2c 0a  ().dates()[0]),.
1220: 09 09 09 09 09 09 09 7d 3f 3b 0a 09 09 09 09 09  .......}?;......
1230: 09 09 6c 65 74 20 75 72 6c 20 3d 20 6c 69 6e 6b  ..let url = link
1240: 3b 0a 09 09 09 09 09 09 09 70 6f 73 74 73 2e 69  ;........posts.i
1250: 6e 73 65 72 74 28 64 61 74 65 2c 20 75 72 6c 2e  nsert(date, url.
1260: 74 6f 5f 73 74 72 69 6e 67 28 29 29 3b 0a 09 09  to_string());...
1270: 09 09 09 09 7d 0a 09 09 09 09 09 7d 3b 0a 09 09  ....}......};...
1280: 09 09 7d 2c 0a 09 09 09 09 45 72 72 28 65 72 72  ..},.....Err(err
1290: 29 20 3d 3e 20 6d 61 74 63 68 20 65 72 72 20 7b  ) => match err {
12a0: 0a 09 09 09 09 09 72 73 73 3a 3a 45 72 72 6f 72  ......rss::Error
12b0: 3a 3a 49 6e 76 61 6c 69 64 53 74 61 72 74 54 61  ::InvalidStartTa
12c0: 67 20 3d 3e 20 7b 0a 09 09 09 09 09 09 6c 65 74  g => {.......let
12d0: 20 66 65 65 64 20 3d 20 61 74 6f 6d 5f 73 79 6e   feed = atom_syn
12e0: 64 69 63 61 74 69 6f 6e 3a 3a 46 65 65 64 3a 3a  dication::Feed::
12f0: 72 65 61 64 5f 66 72 6f 6d 28 26 63 6f 6e 74 65  read_from(&conte
1300: 6e 74 5b 2e 2e 5d 29 0a 09 09 09 09 09 09 09 2e  nt[..]).........
1310: 77 69 74 68 5f 63 6f 6e 74 65 78 74 28 7c 7c 20  with_context(|| 
1320: 66 6f 72 6d 61 74 21 28 22 50 72 6f 62 6c 65 6d  format!("Problem
1330: 20 6f 70 65 6e 69 6e 67 20 66 65 65 64 20 75 72   opening feed ur
1340: 6c 3a 5c 6e 7b 7d 5c 6e 7b 7d 22 2c 20 26 73 6f  l:\n{}\n{}", &so
1350: 75 72 63 65 2e 75 72 6c 2c 20 73 74 61 74 75 73  urce.url, status
1360: 29 29 3f 3b 0a 09 09 09 09 09 09 66 6f 72 20 69  ))?;.......for i
1370: 74 65 6d 20 69 6e 20 66 65 65 64 2e 65 6e 74 72  tem in feed.entr
1380: 69 65 73 28 29 20 7b 0a 09 09 09 09 09 09 09 6c  ies() {........l
1390: 65 74 20 64 61 74 65 20 3d 20 69 74 65 6d 2e 70  et date = item.p
13a0: 75 62 6c 69 73 68 65 64 28 29 2e 75 6e 77 72 61  ublished().unwra
13b0: 70 28 29 3b 0a 09 09 09 09 09 09 09 6c 65 74 20  p();........let 
13c0: 75 72 6c 20 3d 20 69 74 65 6d 2e 6c 69 6e 6b 73  url = item.links
13d0: 28 29 5b 30 5d 2e 68 72 65 66 28 29 3b 0a 09 09  ()[0].href();...
13e0: 09 09 09 09 09 70 6f 73 74 73 2e 69 6e 73 65 72  .....posts.inser
13f0: 74 28 2a 64 61 74 65 2c 20 75 72 6c 2e 74 6f 5f  t(*date, url.to_
1400: 73 74 72 69 6e 67 28 29 29 3b 0a 09 09 09 09 09  string());......
1410: 09 7d 3b 0a 09 09 09 09 09 7d 2c 0a 09 09 09 09  .};......},.....
1420: 09 72 73 73 3a 3a 45 72 72 6f 72 3a 3a 45 6f 66  .rss::Error::Eof
1430: 20 3d 3e 20 28 29 2c 0a 09 09 09 09 09 5f 20 3d   => (),......_ =
1440: 3e 20 62 61 69 6c 21 28 22 55 6e 73 75 70 70 6f  > bail!("Unsuppo
1450: 72 74 65 64 20 6f 72 20 6d 61 6e 67 6c 65 64 20  rted or mangled 
1460: 63 6f 6e 74 65 6e 74 3a 5c 6e 7b 3a 3f 7d 5c 6e  content:\n{:?}\n
1470: 7b 3a 23 3f 7d 5c 6e 7b 3a 23 3f 7d 5c 6e 22 2c  {:#?}\n{:#?}\n",
1480: 20 26 73 6f 75 72 63 65 2e 75 72 6c 2c 20 65 72   &source.url, er
1490: 72 2c 20 73 74 61 74 75 73 29 0a 09 09 09 09 7d  r, status).....}
14a0: 0a 09 09 09 7d 3b 0a 09 09 09 66 6f 72 20 28 64  ....};....for (d
14b0: 61 74 65 2c 20 75 72 6c 29 20 69 6e 20 70 6f 73  ate, url) in pos
14c0: 74 73 2e 69 74 65 72 28 29 20 7b 0a 09 09 09 09  ts.iter() {.....
14d0: 6c 65 74 20 70 6f 73 74 5f 75 72 6c 3a 20 43 6f  let post_url: Co
14e0: 77 3c 73 74 72 3e 20 3d 20 6d 61 74 63 68 20 73  w<str> = match s
14f0: 6f 75 72 63 65 2e 75 72 6c 5f 72 65 20 7b 0a 09  ource.url_re {..
1500: 09 09 09 09 53 6f 6d 65 28 72 65 66 20 78 29 20  ....Some(ref x) 
1510: 3d 3e 20 73 65 64 72 65 67 65 78 3a 3a 52 65 70  => sedregex::Rep
1520: 6c 61 63 65 43 6f 6d 6d 61 6e 64 3a 3a 6e 65 77  laceCommand::new
1530: 28 78 29 3f 2e 65 78 65 63 75 74 65 28 75 72 6c  (x)?.execute(url
1540: 29 2c 0a 09 09 09 09 09 4e 6f 6e 65 20 3d 3e 20  ),......None => 
1550: 75 72 6c 2e 69 6e 74 6f 28 29 2c 0a 09 09 09 09  url.into(),.....
1560: 7d 3b 0a 09 09 09 09 69 66 20 6c 65 74 20 53 6f  };.....if let So
1570: 6d 65 28 65 78 69 73 74 73 29 20 3d 20 73 71 6c  me(exists) = sql
1580: 78 3a 3a 71 75 65 72 79 21 28 22 73 65 6c 65 63  x::query!("selec
1590: 74 20 65 78 69 73 74 73 28 73 65 6c 65 63 74 20  t exists(select 
15a0: 74 72 75 65 20 66 72 6f 6d 20 72 73 73 74 67 5f  true from rsstg_
15b0: 70 6f 73 74 20 77 68 65 72 65 20 75 72 6c 20 3d  post where url =
15c0: 20 24 31 20 61 6e 64 20 73 6f 75 72 63 65 5f 69   $1 and source_i
15d0: 64 20 3d 20 24 32 29 20 61 73 20 65 78 69 73 74  d = $2) as exist
15e0: 73 3b 22 2c 0a 09 09 09 09 09 26 70 6f 73 74 5f  s;",......&post_
15f0: 75 72 6c 2c 20 2a 69 64 29 2e 66 65 74 63 68 5f  url, *id).fetch_
1600: 6f 6e 65 28 26 6d 75 74 20 2a 63 6f 6e 6e 29 2e  one(&mut *conn).
1610: 61 77 61 69 74 3f 2e 65 78 69 73 74 73 20 7b 0a  await?.exists {.
1620: 09 09 09 09 09 69 66 20 21 20 65 78 69 73 74 73  .....if ! exists
1630: 20 7b 0a 09 09 09 09 09 09 69 66 20 74 68 69 73   {.......if this
1640: 5f 66 65 74 63 68 2e 69 73 5f 6e 6f 6e 65 28 29  _fetch.is_none()
1650: 20 7c 7c 20 2a 64 61 74 65 20 3e 20 74 68 69 73   || *date > this
1660: 5f 66 65 74 63 68 2e 75 6e 77 72 61 70 28 29 20  _fetch.unwrap() 
1670: 7b 0a 09 09 09 09 09 09 09 74 68 69 73 5f 66 65  {........this_fe
1680: 74 63 68 20 3d 20 53 6f 6d 65 28 2a 64 61 74 65  tch = Some(*date
1690: 29 3b 0a 09 09 09 09 09 09 7d 3b 0a 09 09 09 09  );.......};.....
16a0: 09 09 73 65 6c 66 2e 72 65 71 75 65 73 74 28 20  ..self.request( 
16b0: 6d 61 74 63 68 20 26 73 6f 75 72 63 65 2e 69 76  match &source.iv
16c0: 5f 68 61 73 68 20 7b 0a 09 09 09 09 09 09 09 09  _hash {.........
16d0: 53 6f 6d 65 28 68 61 73 68 29 20 3d 3e 20 74 65  Some(hash) => te
16e0: 6c 65 67 72 61 6d 5f 62 6f 74 3a 3a 53 65 6e 64  legram_bot::Send
16f0: 4d 65 73 73 61 67 65 3a 3a 6e 65 77 28 64 65 73  Message::new(des
1700: 74 69 6e 61 74 69 6f 6e 2c 20 66 6f 72 6d 61 74  tination, format
1710: 21 28 22 3c 61 20 68 72 65 66 3d 5c 22 68 74 74  !("<a href=\"htt
1720: 70 73 3a 2f 2f 74 2e 6d 65 2f 69 76 3f 75 72 6c  ps://t.me/iv?url
1730: 3d 7b 7d 26 72 68 61 73 68 3d 7b 7d 5c 22 3e 20  ={}&rhash={}\"> 
1740: 3c 2f 61 3e 7b 30 7d 22 2c 20 26 70 6f 73 74 5f  </a>{0}", &post_
1750: 75 72 6c 2c 20 68 61 73 68 29 29 2c 0a 09 09 09  url, hash)),....
1760: 09 09 09 09 09 4e 6f 6e 65 20 3d 3e 20 74 65 6c  .....None => tel
1770: 65 67 72 61 6d 5f 62 6f 74 3a 3a 53 65 6e 64 4d  egram_bot::SendM
1780: 65 73 73 61 67 65 3a 3a 6e 65 77 28 64 65 73 74  essage::new(dest
1790: 69 6e 61 74 69 6f 6e 2c 20 66 6f 72 6d 61 74 21  ination, format!
17a0: 28 22 7b 7d 22 2c 20 70 6f 73 74 5f 75 72 6c 29  ("{}", post_url)
17b0: 29 2c 0a 09 09 09 09 09 09 09 7d 2e 70 61 72 73  ),........}.pars
17c0: 65 5f 6d 6f 64 65 28 74 65 6c 65 67 72 61 6d 5f  e_mode(telegram_
17d0: 62 6f 74 3a 3a 74 79 70 65 73 3a 3a 50 61 72 73  bot::types::Pars
17e0: 65 4d 6f 64 65 3a 3a 48 74 6d 6c 29 29 2e 61 77  eMode::Html)).aw
17f0: 61 69 74 0a 09 09 09 09 09 09 09 2e 63 6f 6e 74  ait.........cont
1800: 65 78 74 28 22 43 61 6e 27 74 20 70 6f 73 74 20  ext("Can't post 
1810: 6d 65 73 73 61 67 65 3a 22 29 3f 3b 0a 09 09 09  message:")?;....
1820: 09 09 09 73 71 6c 78 3a 3a 71 75 65 72 79 21 28  ...sqlx::query!(
1830: 22 69 6e 73 65 72 74 20 69 6e 74 6f 20 72 73 73  "insert into rss
1840: 74 67 5f 70 6f 73 74 20 28 73 6f 75 72 63 65 5f  tg_post (source_
1850: 69 64 2c 20 70 6f 73 74 65 64 2c 20 75 72 6c 29  id, posted, url)
1860: 20 76 61 6c 75 65 73 20 28 24 31 2c 20 24 32 2c   values ($1, $2,
1870: 20 24 33 29 3b 22 2c 0a 09 09 09 09 09 09 09 2a   $3);",........*
1880: 69 64 2c 20 64 61 74 65 2c 20 26 70 6f 73 74 5f  id, date, &post_
1890: 75 72 6c 29 2e 65 78 65 63 75 74 65 28 26 6d 75  url).execute(&mu
18a0: 74 20 2a 63 6f 6e 6e 29 2e 61 77 61 69 74 3f 3b  t *conn).await?;
18b0: 0a 09 09 09 09 09 7d 3b 0a 09 09 09 09 7d 3b 0a  ......};.....};.
18c0: 09 09 09 09 70 6f 73 74 65 64 20 2b 3d 20 31 3b  ....posted += 1;
18d0: 0a 09 09 09 7d 3b 0a 09 09 09 70 6f 73 74 73 2e  ....};....posts.
18e0: 63 6c 65 61 72 28 29 3b 0a 09 09 7d 3b 0a 09 09  clear();...};...
18f0: 73 71 6c 78 3a 3a 71 75 65 72 79 21 28 22 75 70  sqlx::query!("up
1900: 64 61 74 65 20 72 73 73 74 67 5f 73 6f 75 72 63  date rsstg_sourc
1910: 65 20 73 65 74 20 6c 61 73 74 5f 73 63 72 61 70  e set last_scrap
1920: 65 20 3d 20 6e 6f 77 28 29 20 77 68 65 72 65 20  e = now() where 
1930: 73 6f 75 72 63 65 5f 69 64 20 3d 20 24 31 3b 22  source_id = $1;"
1940: 2c 0a 09 09 09 2a 69 64 29 2e 65 78 65 63 75 74  ,....*id).execut
1950: 65 28 26 6d 75 74 20 2a 63 6f 6e 6e 29 2e 61 77  e(&mut *conn).aw
1960: 61 69 74 3f 3b 0a 09 09 4f 6b 28 66 6f 72 6d 61  ait?;...Ok(forma
1970: 74 21 28 22 50 6f 73 74 65 64 3a 20 7b 7d 22 2c  t!("Posted: {}",
1980: 20 26 70 6f 73 74 65 64 29 2e 69 6e 74 6f 28 29   &posted).into()
1990: 29 0a 09 7d 0a 0a 09 70 75 62 20 61 73 79 6e 63  )..}...pub async
19a0: 20 66 6e 20 64 65 6c 65 74 65 3c 53 3e 28 26 73   fn delete<S>(&s
19b0: 65 6c 66 2c 20 73 6f 75 72 63 65 5f 69 64 3a 20  elf, source_id: 
19c0: 26 69 33 32 2c 20 6f 77 6e 65 72 3a 20 53 29 20  &i32, owner: S) 
19d0: 2d 3e 20 52 65 73 75 6c 74 3c 43 6f 77 3c 27 5f  -> Result<Cow<'_
19e0: 2c 20 73 74 72 3e 3e 0a 09 77 68 65 72 65 20 53  , str>>..where S
19f0: 3a 20 49 6e 74 6f 3c 69 36 34 3e 20 7b 0a 09 09  : Into<i64> {...
1a00: 6c 65 74 20 6f 77 6e 65 72 20 3d 20 6f 77 6e 65  let owner = owne
1a10: 72 2e 69 6e 74 6f 28 29 3b 0a 0a 09 09 6d 61 74  r.into();....mat
1a20: 63 68 20 73 71 6c 78 3a 3a 71 75 65 72 79 21 28  ch sqlx::query!(
1a30: 22 64 65 6c 65 74 65 20 66 72 6f 6d 20 72 73 73  "delete from rss
1a40: 74 67 5f 73 6f 75 72 63 65 20 77 68 65 72 65 20  tg_source where 
1a50: 73 6f 75 72 63 65 5f 69 64 20 3d 20 24 31 20 61  source_id = $1 a
1a60: 6e 64 20 6f 77 6e 65 72 20 3d 20 24 32 3b 22 2c  nd owner = $2;",
1a70: 0a 09 09 09 73 6f 75 72 63 65 5f 69 64 2c 20 6f  ....source_id, o
1a80: 77 6e 65 72 29 2e 65 78 65 63 75 74 65 28 26 6d  wner).execute(&m
1a90: 75 74 20 2a 73 65 6c 66 2e 70 6f 6f 6c 2e 61 63  ut *self.pool.ac
1aa0: 71 75 69 72 65 28 29 2e 61 77 61 69 74 3f 29 2e  quire().await?).
1ab0: 61 77 61 69 74 3f 2e 72 6f 77 73 5f 61 66 66 65  await?.rows_affe
1ac0: 63 74 65 64 28 29 20 7b 0a 09 09 09 30 20 3d 3e  cted() {....0 =>
1ad0: 20 7b 20 4f 6b 28 22 4e 6f 20 64 61 74 61 20 66   { Ok("No data f
1ae0: 6f 75 6e 64 20 66 6f 75 6e 64 2e 22 2e 69 6e 74  ound found.".int
1af0: 6f 28 29 29 20 7d 2c 0a 09 09 09 78 20 3d 3e 20  o()) },....x => 
1b00: 7b 20 4f 6b 28 66 6f 72 6d 61 74 21 28 22 7b 7d  { Ok(format!("{}
1b10: 20 73 6f 75 72 63 65 73 20 72 65 6d 6f 76 65 64   sources removed
1b20: 2e 22 2c 20 78 29 2e 69 6e 74 6f 28 29 29 20 7d  .", x).into()) }
1b30: 2c 0a 09 09 7d 0a 09 7d 0a 0a 09 70 75 62 20 61  ,...}..}...pub a
1b40: 73 79 6e 63 20 66 6e 20 63 6c 65 61 6e 3c 53 3e  sync fn clean<S>
1b50: 28 26 73 65 6c 66 2c 20 73 6f 75 72 63 65 5f 69  (&self, source_i
1b60: 64 3a 20 26 69 33 32 2c 20 6f 77 6e 65 72 3a 20  d: &i32, owner: 
1b70: 53 29 20 2d 3e 20 52 65 73 75 6c 74 3c 43 6f 77  S) -> Result<Cow
1b80: 3c 27 5f 2c 20 73 74 72 3e 3e 0a 09 77 68 65 72  <'_, str>>..wher
1b90: 65 20 53 3a 20 49 6e 74 6f 3c 69 36 34 3e 20 7b  e S: Into<i64> {
1ba0: 0a 09 09 6c 65 74 20 6f 77 6e 65 72 20 3d 20 6f  ...let owner = o
1bb0: 77 6e 65 72 2e 69 6e 74 6f 28 29 3b 0a 0a 09 09  wner.into();....
1bc0: 6d 61 74 63 68 20 73 71 6c 78 3a 3a 71 75 65 72  match sqlx::quer
1bd0: 79 21 28 22 64 65 6c 65 74 65 20 66 72 6f 6d 20  y!("delete from 
1be0: 72 73 73 74 67 5f 70 6f 73 74 20 70 20 75 73 69  rsstg_post p usi
1bf0: 6e 67 20 72 73 73 74 67 5f 73 6f 75 72 63 65 20  ng rsstg_source 
1c00: 73 20 77 68 65 72 65 20 70 2e 73 6f 75 72 63 65  s where p.source
1c10: 5f 69 64 20 3d 20 24 31 20 61 6e 64 20 6f 77 6e  _id = $1 and own
1c20: 65 72 20 3d 20 24 32 20 61 6e 64 20 70 2e 73 6f  er = $2 and p.so
1c30: 75 72 63 65 5f 69 64 20 3d 20 73 2e 73 6f 75 72  urce_id = s.sour
1c40: 63 65 5f 69 64 3b 22 2c 0a 09 09 09 73 6f 75 72  ce_id;",....sour
1c50: 63 65 5f 69 64 2c 20 6f 77 6e 65 72 29 2e 65 78  ce_id, owner).ex
1c60: 65 63 75 74 65 28 26 6d 75 74 20 2a 73 65 6c 66  ecute(&mut *self
1c70: 2e 70 6f 6f 6c 2e 61 63 71 75 69 72 65 28 29 2e  .pool.acquire().
1c80: 61 77 61 69 74 3f 29 2e 61 77 61 69 74 3f 2e 72  await?).await?.r
1c90: 6f 77 73 5f 61 66 66 65 63 74 65 64 28 29 20 7b  ows_affected() {
1ca0: 0a 09 09 09 30 20 3d 3e 20 7b 20 4f 6b 28 22 4e  ....0 => { Ok("N
1cb0: 6f 20 64 61 74 61 20 66 6f 75 6e 64 20 66 6f 75  o data found fou
1cc0: 6e 64 2e 22 2e 69 6e 74 6f 28 29 29 20 7d 2c 0a  nd.".into()) },.
1cd0: 09 09 09 78 20 3d 3e 20 7b 20 4f 6b 28 66 6f 72  ...x => { Ok(for
1ce0: 6d 61 74 21 28 22 7b 7d 20 70 6f 73 74 73 20 70  mat!("{} posts p
1cf0: 75 72 67 65 64 2e 22 2c 20 78 29 2e 69 6e 74 6f  urged.", x).into
1d00: 28 29 29 20 7d 2c 0a 09 09 7d 0a 09 7d 0a 0a 09  ()) },...}..}...
1d10: 70 75 62 20 61 73 79 6e 63 20 66 6e 20 65 6e 61  pub async fn ena
1d20: 62 6c 65 3c 53 3e 28 26 73 65 6c 66 2c 20 73 6f  ble<S>(&self, so
1d30: 75 72 63 65 5f 69 64 3a 20 26 69 33 32 2c 20 6f  urce_id: &i32, o
1d40: 77 6e 65 72 3a 20 53 29 20 2d 3e 20 52 65 73 75  wner: S) -> Resu
1d50: 6c 74 3c 26 73 74 72 3e 0a 09 77 68 65 72 65 20  lt<&str>..where 
1d60: 53 3a 20 49 6e 74 6f 3c 69 36 34 3e 20 7b 0a 09  S: Into<i64> {..
1d70: 09 6c 65 74 20 6f 77 6e 65 72 20 3d 20 6f 77 6e  .let owner = own
1d80: 65 72 2e 69 6e 74 6f 28 29 3b 0a 0a 09 09 6d 61  er.into();....ma
1d90: 74 63 68 20 73 71 6c 78 3a 3a 71 75 65 72 79 21  tch sqlx::query!
1da0: 28 22 75 70 64 61 74 65 20 72 73 73 74 67 5f 73  ("update rsstg_s
1db0: 6f 75 72 63 65 20 73 65 74 20 65 6e 61 62 6c 65  ource set enable
1dc0: 64 20 3d 20 74 72 75 65 20 77 68 65 72 65 20 73  d = true where s
1dd0: 6f 75 72 63 65 5f 69 64 20 3d 20 24 31 20 61 6e  ource_id = $1 an
1de0: 64 20 6f 77 6e 65 72 20 3d 20 24 32 22 2c 0a 09  d owner = $2",..
1df0: 09 09 73 6f 75 72 63 65 5f 69 64 2c 20 6f 77 6e  ..source_id, own
1e00: 65 72 29 2e 65 78 65 63 75 74 65 28 26 6d 75 74  er).execute(&mut
1e10: 20 2a 73 65 6c 66 2e 70 6f 6f 6c 2e 61 63 71 75   *self.pool.acqu
1e20: 69 72 65 28 29 2e 61 77 61 69 74 3f 29 2e 61 77  ire().await?).aw
1e30: 61 69 74 3f 2e 72 6f 77 73 5f 61 66 66 65 63 74  ait?.rows_affect
1e40: 65 64 28 29 20 7b 0a 09 09 09 31 20 3d 3e 20 7b  ed() {....1 => {
1e50: 20 4f 6b 28 22 53 6f 75 72 63 65 20 65 6e 61 62   Ok("Source enab
1e60: 6c 65 64 2e 22 29 20 7d 2c 0a 09 09 09 30 20 3d  led.") },....0 =
1e70: 3e 20 7b 20 4f 6b 28 22 53 6f 75 72 63 65 20 6e  > { Ok("Source n
1e80: 6f 74 20 66 6f 75 6e 64 2e 22 29 20 7d 2c 0a 09  ot found.") },..
1e90: 09 09 5f 20 3d 3e 20 7b 20 45 72 72 28 61 6e 79  .._ => { Err(any
1ea0: 68 6f 77 21 28 22 44 61 74 61 62 61 73 65 20 65  how!("Database e
1eb0: 72 72 6f 72 2e 22 29 29 20 7d 2c 0a 09 09 7d 0a  rror.")) },...}.
1ec0: 09 7d 0a 0a 09 70 75 62 20 61 73 79 6e 63 20 66  .}...pub async f
1ed0: 6e 20 64 69 73 61 62 6c 65 3c 53 3e 28 26 73 65  n disable<S>(&se
1ee0: 6c 66 2c 20 73 6f 75 72 63 65 5f 69 64 3a 20 26  lf, source_id: &
1ef0: 69 33 32 2c 20 6f 77 6e 65 72 3a 20 53 29 20 2d  i32, owner: S) -
1f00: 3e 20 52 65 73 75 6c 74 3c 26 73 74 72 3e 0a 09  > Result<&str>..
1f10: 77 68 65 72 65 20 53 3a 20 49 6e 74 6f 3c 69 36  where S: Into<i6
1f20: 34 3e 20 7b 0a 09 09 6c 65 74 20 6f 77 6e 65 72  4> {...let owner
1f30: 20 3d 20 6f 77 6e 65 72 2e 69 6e 74 6f 28 29 3b   = owner.into();
1f40: 0a 0a 09 09 6d 61 74 63 68 20 73 71 6c 78 3a 3a  ....match sqlx::
1f50: 71 75 65 72 79 21 28 22 75 70 64 61 74 65 20 72  query!("update r
1f60: 73 73 74 67 5f 73 6f 75 72 63 65 20 73 65 74 20  sstg_source set 
1f70: 65 6e 61 62 6c 65 64 20 3d 20 66 61 6c 73 65 20  enabled = false 
1f80: 77 68 65 72 65 20 73 6f 75 72 63 65 5f 69 64 20  where source_id 
1f90: 3d 20 24 31 20 61 6e 64 20 6f 77 6e 65 72 20 3d  = $1 and owner =
1fa0: 20 24 32 22 2c 0a 09 09 09 73 6f 75 72 63 65 5f   $2",....source_
1fb0: 69 64 2c 20 6f 77 6e 65 72 29 2e 65 78 65 63 75  id, owner).execu
1fc0: 74 65 28 26 6d 75 74 20 2a 73 65 6c 66 2e 70 6f  te(&mut *self.po
1fd0: 6f 6c 2e 61 63 71 75 69 72 65 28 29 2e 61 77 61  ol.acquire().awa
1fe0: 69 74 3f 29 2e 61 77 61 69 74 3f 2e 72 6f 77 73  it?).await?.rows
1ff0: 5f 61 66 66 65 63 74 65 64 28 29 20 7b 0a 09 09  _affected() {...
2000: 09 31 20 3d 3e 20 7b 20 4f 6b 28 22 53 6f 75 72  .1 => { Ok("Sour
2010: 63 65 20 64 69 73 61 62 6c 65 64 2e 22 29 20 7d  ce disabled.") }
2020: 2c 0a 09 09 09 30 20 3d 3e 20 7b 20 4f 6b 28 22  ,....0 => { Ok("
2030: 53 6f 75 72 63 65 20 6e 6f 74 20 66 6f 75 6e 64  Source not found
2040: 2e 22 29 20 7d 2c 0a 09 09 09 5f 20 3d 3e 20 7b  .") },...._ => {
2050: 20 45 72 72 28 61 6e 79 68 6f 77 21 28 22 44 61   Err(anyhow!("Da
2060: 74 61 62 61 73 65 20 65 72 72 6f 72 2e 22 29 29  tabase error."))
2070: 20 7d 2c 0a 09 09 7d 0a 09 7d 0a 0a 09 70 75 62   },...}..}...pub
2080: 20 61 73 79 6e 63 20 66 6e 20 75 70 64 61 74 65   async fn update
2090: 3c 53 3e 28 26 73 65 6c 66 2c 20 75 70 64 61 74  <S>(&self, updat
20a0: 65 3a 20 4f 70 74 69 6f 6e 3c 69 33 32 3e 2c 20  e: Option<i32>, 
20b0: 63 68 61 6e 6e 65 6c 3a 20 26 73 74 72 2c 20 63  channel: &str, c
20c0: 68 61 6e 6e 65 6c 5f 69 64 3a 20 69 36 34 2c 20  hannel_id: i64, 
20d0: 75 72 6c 3a 20 26 73 74 72 2c 20 69 76 5f 68 61  url: &str, iv_ha
20e0: 73 68 3a 20 4f 70 74 69 6f 6e 3c 26 73 74 72 3e  sh: Option<&str>
20f0: 2c 20 75 72 6c 5f 72 65 3a 20 4f 70 74 69 6f 6e  , url_re: Option
2100: 3c 26 73 74 72 3e 2c 20 6f 77 6e 65 72 3a 20 53  <&str>, owner: S
2110: 29 20 2d 3e 20 52 65 73 75 6c 74 3c 26 73 74 72  ) -> Result<&str
2120: 3e 0a 09 77 68 65 72 65 20 53 3a 20 49 6e 74 6f  >..where S: Into
2130: 3c 69 36 34 3e 20 7b 0a 09 09 6c 65 74 20 6f 77  <i64> {...let ow
2140: 6e 65 72 20 3d 20 6f 77 6e 65 72 2e 69 6e 74 6f  ner = owner.into
2150: 28 29 3b 0a 09 09 6c 65 74 20 6d 75 74 20 63 6f  ();...let mut co
2160: 6e 6e 20 3d 20 73 65 6c 66 2e 70 6f 6f 6c 2e 61  nn = self.pool.a
2170: 63 71 75 69 72 65 28 29 2e 61 77 61 69 74 3f 3b  cquire().await?;
2180: 0a 0a 09 09 6d 61 74 63 68 20 6d 61 74 63 68 20  ....match match 
2190: 75 70 64 61 74 65 20 7b 0a 09 09 09 09 53 6f 6d  update {.....Som
21a0: 65 28 69 64 29 20 3d 3e 20 7b 0a 09 09 09 09 09  e(id) => {......
21b0: 73 71 6c 78 3a 3a 71 75 65 72 79 21 28 22 75 70  sqlx::query!("up
21c0: 64 61 74 65 20 72 73 73 74 67 5f 73 6f 75 72 63  date rsstg_sourc
21d0: 65 20 73 65 74 20 63 68 61 6e 6e 65 6c 5f 69 64  e set channel_id
21e0: 20 3d 20 24 32 2c 20 75 72 6c 20 3d 20 24 33 2c   = $2, url = $3,
21f0: 20 69 76 5f 68 61 73 68 20 3d 20 24 34 2c 20 6f   iv_hash = $4, o
2200: 77 6e 65 72 20 3d 20 24 35 2c 20 63 68 61 6e 6e  wner = $5, chann
2210: 65 6c 20 3d 20 24 36 2c 20 75 72 6c 5f 72 65 20  el = $6, url_re 
2220: 3d 20 24 37 20 77 68 65 72 65 20 73 6f 75 72 63  = $7 where sourc
2230: 65 5f 69 64 20 3d 20 24 31 22 2c 0a 09 09 09 09  e_id = $1",.....
2240: 09 09 69 64 2c 20 63 68 61 6e 6e 65 6c 5f 69 64  ..id, channel_id
2250: 2c 20 75 72 6c 2c 20 69 76 5f 68 61 73 68 2c 20  , url, iv_hash, 
2260: 6f 77 6e 65 72 2c 20 63 68 61 6e 6e 65 6c 2c 20  owner, channel, 
2270: 75 72 6c 5f 72 65 29 2e 65 78 65 63 75 74 65 28  url_re).execute(
2280: 26 6d 75 74 20 2a 63 6f 6e 6e 29 2e 61 77 61 69  &mut *conn).awai
2290: 74 0a 09 09 09 09 7d 2c 0a 09 09 09 09 4e 6f 6e  t.....},.....Non
22a0: 65 20 3d 3e 20 7b 0a 09 09 09 09 09 73 71 6c 78  e => {......sqlx
22b0: 3a 3a 71 75 65 72 79 21 28 22 69 6e 73 65 72 74  ::query!("insert
22c0: 20 69 6e 74 6f 20 72 73 73 74 67 5f 73 6f 75 72   into rsstg_sour
22d0: 63 65 20 28 63 68 61 6e 6e 65 6c 5f 69 64 2c 20  ce (channel_id, 
22e0: 75 72 6c 2c 20 69 76 5f 68 61 73 68 2c 20 6f 77  url, iv_hash, ow
22f0: 6e 65 72 2c 20 63 68 61 6e 6e 65 6c 2c 20 75 72  ner, channel, ur
2300: 6c 5f 72 65 29 20 76 61 6c 75 65 73 20 28 24 31  l_re) values ($1
2310: 2c 20 24 32 2c 20 24 33 2c 20 24 34 2c 20 24 35  , $2, $3, $4, $5
2320: 2c 20 24 36 29 22 2c 0a 09 09 09 09 09 09 63 68  , $6)",.......ch
2330: 61 6e 6e 65 6c 5f 69 64 2c 20 75 72 6c 2c 20 69  annel_id, url, i
2340: 76 5f 68 61 73 68 2c 20 6f 77 6e 65 72 2c 20 63  v_hash, owner, c
2350: 68 61 6e 6e 65 6c 2c 20 75 72 6c 5f 72 65 29 2e  hannel, url_re).
2360: 65 78 65 63 75 74 65 28 26 6d 75 74 20 2a 63 6f  execute(&mut *co
2370: 6e 6e 29 2e 61 77 61 69 74 0a 09 09 09 09 7d 2c  nn).await.....},
2380: 0a 09 09 09 7d 20 7b 0a 09 09 09 4f 6b 28 5f 29  ....} {....Ok(_)
2390: 20 3d 3e 20 4f 6b 28 6d 61 74 63 68 20 75 70 64   => Ok(match upd
23a0: 61 74 65 20 7b 0a 09 09 09 09 53 6f 6d 65 28 5f  ate {.....Some(_
23b0: 29 20 3d 3e 20 22 43 68 61 6e 6e 65 6c 20 75 70  ) => "Channel up
23c0: 64 61 74 65 64 2e 22 2c 0a 09 09 09 09 4e 6f 6e  dated.",.....Non
23d0: 65 20 3d 3e 20 22 43 68 61 6e 6e 65 6c 20 61 64  e => "Channel ad
23e0: 64 65 64 2e 22 2c 0a 09 09 09 7d 29 2c 0a 09 09  ded.",....}),...
23f0: 09 45 72 72 28 73 71 6c 78 3a 3a 45 72 72 6f 72  .Err(sqlx::Error
2400: 3a 3a 44 61 74 61 62 61 73 65 28 65 72 72 29 29  ::Database(err))
2410: 20 3d 3e 20 7b 0a 09 09 09 09 6d 61 74 63 68 20   => {.....match 
2420: 65 72 72 2e 64 6f 77 6e 63 61 73 74 3a 3a 3c 73  err.downcast::<s
2430: 71 6c 78 3a 3a 70 6f 73 74 67 72 65 73 3a 3a 50  qlx::postgres::P
2440: 67 44 61 74 61 62 61 73 65 45 72 72 6f 72 3e 28  gDatabaseError>(
2450: 29 2e 72 6f 75 74 69 6e 65 28 29 20 7b 0a 09 09  ).routine() {...
2460: 09 09 09 53 6f 6d 65 28 22 5f 62 74 5f 63 68 65  ...Some("_bt_che
2470: 63 6b 5f 75 6e 69 71 75 65 22 2c 20 29 20 3d 3e  ck_unique", ) =>
2480: 20 7b 0a 09 09 09 09 09 09 4f 6b 28 22 44 75 70   {.......Ok("Dup
2490: 6c 69 63 61 74 65 20 6b 65 79 2e 22 29 0a 09 09  licate key.")...
24a0: 09 09 09 7d 2c 0a 09 09 09 09 09 53 6f 6d 65 28  ...},......Some(
24b0: 5f 29 20 3d 3e 20 7b 0a 09 09 09 09 09 09 4f 6b  _) => {.......Ok
24c0: 28 22 44 61 74 61 62 61 73 65 20 65 72 72 6f 72  ("Database error
24d0: 2e 22 29 0a 09 09 09 09 09 7d 2c 0a 09 09 09 09  .")......},.....
24e0: 09 4e 6f 6e 65 20 3d 3e 20 7b 0a 09 09 09 09 09  .None => {......
24f0: 09 4f 6b 28 22 4e 6f 20 64 61 74 61 62 61 73 65  .Ok("No database
2500: 20 65 72 72 6f 72 20 65 78 74 72 61 63 74 65 64   error extracted
2510: 2e 22 29 0a 09 09 09 09 09 7d 2c 0a 09 09 09 09  .")......},.....
2520: 7d 0a 09 09 09 7d 2c 0a 09 09 09 45 72 72 28 65  }....},....Err(e
2530: 72 72 29 20 3d 3e 20 7b 0a 09 09 09 09 62 61 69  rr) => {.....bai
2540: 6c 21 28 22 53 6f 72 72 79 2c 20 75 6e 6b 6e 6f  l!("Sorry, unkno
2550: 77 6e 20 65 72 72 6f 72 3a 5c 6e 7b 3a 23 3f 7d  wn error:\n{:#?}
2560: 5c 6e 22 2c 20 65 72 72 29 3b 0a 09 09 09 7d 2c  \n", err);....},
2570: 0a 09 09 7d 0a 09 7d 0a 0a 09 61 73 79 6e 63 20  ...}..}...async 
2580: 66 6e 20 61 75 74 6f 66 65 74 63 68 28 26 73 65  fn autofetch(&se
2590: 6c 66 29 20 2d 3e 20 52 65 73 75 6c 74 3c 73 74  lf) -> Result<st
25a0: 64 3a 3a 74 69 6d 65 3a 3a 44 75 72 61 74 69 6f  d::time::Duratio
25b0: 6e 3e 20 7b 0a 09 09 6c 65 74 20 6d 75 74 20 64  n> {...let mut d
25c0: 65 6c 61 79 20 3d 20 63 68 72 6f 6e 6f 3a 3a 44  elay = chrono::D
25d0: 75 72 61 74 69 6f 6e 3a 3a 6d 69 6e 75 74 65 73  uration::minutes
25e0: 28 31 29 3b 0a 09 09 6c 65 74 20 6e 6f 77 20 3d  (1);...let now =
25f0: 20 63 68 72 6f 6e 6f 3a 3a 4c 6f 63 61 6c 3a 3a   chrono::Local::
2600: 6e 6f 77 28 29 3b 0a 09 09 6c 65 74 20 6d 75 74  now();...let mut
2610: 20 71 75 65 75 65 20 3d 20 73 71 6c 78 3a 3a 71   queue = sqlx::q
2620: 75 65 72 79 21 28 72 23 22 73 65 6c 65 63 74 20  uery!(r#"select 
2630: 73 6f 75 72 63 65 5f 69 64 2c 20 6e 65 78 74 5f  source_id, next_
2640: 66 65 74 63 68 20 61 73 20 22 6e 65 78 74 5f 66  fetch as "next_f
2650: 65 74 63 68 3a 20 44 61 74 65 54 69 6d 65 3c 63  etch: DateTime<c
2660: 68 72 6f 6e 6f 3a 3a 4c 6f 63 61 6c 3e 22 2c 20  hrono::Local>", 
2670: 6f 77 6e 65 72 20 66 72 6f 6d 20 72 73 73 74 67  owner from rsstg
2680: 5f 6f 72 64 65 72 20 6e 61 74 75 72 61 6c 20 6c  _order natural l
2690: 65 66 74 20 6a 6f 69 6e 20 72 73 73 74 67 5f 73  eft join rsstg_s
26a0: 6f 75 72 63 65 20 77 68 65 72 65 20 6e 65 78 74  ource where next
26b0: 5f 66 65 74 63 68 20 3c 20 6e 6f 77 28 29 20 2b  _fetch < now() +
26c0: 20 69 6e 74 65 72 76 61 6c 20 27 31 20 6d 69 6e   interval '1 min
26d0: 75 74 65 27 3b 22 23 29 0a 09 09 09 2e 66 65 74  ute';"#).....fet
26e0: 63 68 5f 61 6c 6c 28 26 6d 75 74 20 2a 73 65 6c  ch_all(&mut *sel
26f0: 66 2e 70 6f 6f 6c 2e 61 63 71 75 69 72 65 28 29  f.pool.acquire()
2700: 2e 61 77 61 69 74 3f 29 2e 61 77 61 69 74 3f 3b  .await?).await?;
2710: 0a 09 09 66 6f 72 20 72 6f 77 20 69 6e 20 71 75  ...for row in qu
2720: 65 75 65 2e 69 74 65 72 28 29 20 7b 0a 09 09 09  eue.iter() {....
2730: 69 66 20 6c 65 74 20 53 6f 6d 65 28 6e 65 78 74  if let Some(next
2740: 5f 66 65 74 63 68 29 20 3d 20 72 6f 77 2e 6e 65  _fetch) = row.ne
2750: 78 74 5f 66 65 74 63 68 20 7b 0a 09 09 09 09 69  xt_fetch {.....i
2760: 66 20 6e 65 78 74 5f 66 65 74 63 68 20 3c 20 6e  f next_fetch < n
2770: 6f 77 20 7b 0a 09 09 09 09 09 69 66 20 6c 65 74  ow {......if let
2780: 20 28 53 6f 6d 65 28 6f 77 6e 65 72 29 2c 20 53   (Some(owner), S
2790: 6f 6d 65 28 73 6f 75 72 63 65 5f 69 64 29 29 20  ome(source_id)) 
27a0: 3d 20 28 72 6f 77 2e 6f 77 6e 65 72 2c 20 72 6f  = (row.owner, ro
27b0: 77 2e 73 6f 75 72 63 65 5f 69 64 29 20 7b 0a 09  w.source_id) {..
27c0: 09 09 09 09 09 6c 65 74 20 63 6c 6f 6e 65 20 3d  .....let clone =
27d0: 20 43 6f 72 65 20 7b 0a 09 09 09 09 09 09 09 6f   Core {........o
27e0: 77 6e 65 72 5f 63 68 61 74 3a 20 74 65 6c 65 67  wner_chat: teleg
27f0: 72 61 6d 5f 62 6f 74 3a 3a 55 73 65 72 49 64 3a  ram_bot::UserId:
2800: 3a 6e 65 77 28 6f 77 6e 65 72 29 2c 0a 09 09 09  :new(owner),....
2810: 09 09 09 09 2e 2e 73 65 6c 66 2e 63 6c 6f 6e 65  ......self.clone
2820: 28 29 0a 09 09 09 09 09 09 7d 3b 0a 09 09 09 09  ().......};.....
2830: 09 09 74 61 73 6b 3a 3a 73 70 61 77 6e 28 61 73  ..task::spawn(as
2840: 79 6e 63 20 6d 6f 76 65 20 7b 0a 09 09 09 09 09  ync move {......
2850: 09 09 69 66 20 6c 65 74 20 45 72 72 28 65 72 72  ..if let Err(err
2860: 29 20 3d 20 63 6c 6f 6e 65 2e 63 68 65 63 6b 28  ) = clone.check(
2870: 26 73 6f 75 72 63 65 5f 69 64 2c 20 6f 77 6e 65  &source_id, owne
2880: 72 2c 20 74 72 75 65 29 2e 61 77 61 69 74 20 7b  r, true).await {
2890: 0a 09 09 09 09 09 09 09 09 69 66 20 6c 65 74 20  .........if let 
28a0: 45 72 72 28 65 72 72 29 20 3d 20 63 6c 6f 6e 65  Err(err) = clone
28b0: 2e 73 65 6e 64 28 26 66 6f 72 6d 61 74 21 28 22  .send(&format!("
28c0: f0 9f 9b 91 20 7b 3a 3f 7d 22 2c 20 65 72 72 29  šŸ›‘ {:?}", err)
28d0: 2c 20 4e 6f 6e 65 2c 20 4e 6f 6e 65 29 2e 61 77  , None, None).aw
28e0: 61 69 74 20 7b 0a 09 09 09 09 09 09 09 09 09 64  ait {..........d
28f0: 62 67 21 28 22 43 68 65 63 6b 20 65 72 72 6f 72  bg!("Check error
2900: 3a 20 7b 7d 22 2c 20 65 72 72 29 3b 0a 09 09 09  : {}", err);....
2910: 09 09 09 09 09 09 2f 2f 20 63 6c 6f 6e 65 2e 64  ......// clone.d
2920: 69 73 61 62 6c 65 28 26 73 6f 75 72 63 65 5f 69  isable(&source_i
2930: 64 2c 20 6f 77 6e 65 72 29 2e 61 77 61 69 74 2e  d, owner).await.
2940: 75 6e 77 72 61 70 28 29 3b 0a 09 09 09 09 09 09  unwrap();.......
2950: 09 09 7d 3b 0a 09 09 09 09 09 09 09 7d 3b 0a 09  ..};........};..
2960: 09 09 09 09 09 7d 29 3b 0a 09 09 09 09 09 7d 0a  .....});......}.
2970: 09 09 09 09 7d 20 65 6c 73 65 20 69 66 20 6e 65  ....} else if ne
2980: 78 74 5f 66 65 74 63 68 20 2d 20 6e 6f 77 20 3c  xt_fetch - now <
2990: 20 64 65 6c 61 79 20 7b 0a 09 09 09 09 09 64 65   delay {......de
29a0: 6c 61 79 20 3d 20 6e 65 78 74 5f 66 65 74 63 68  lay = next_fetch
29b0: 20 2d 20 6e 6f 77 3b 0a 09 09 09 09 7d 0a 09 09   - now;.....}...
29c0: 09 7d 0a 09 09 7d 3b 0a 09 09 71 75 65 75 65 2e  .}...};...queue.
29d0: 63 6c 65 61 72 28 29 3b 0a 09 09 4f 6b 28 64 65  clear();...Ok(de
29e0: 6c 61 79 2e 74 6f 5f 73 74 64 28 29 3f 29 0a 09  lay.to_std()?)..
29f0: 7d 0a 0a 09 70 75 62 20 61 73 79 6e 63 20 66 6e  }...pub async fn
2a00: 20 6c 69 73 74 3c 53 3e 28 26 73 65 6c 66 2c 20   list<S>(&self, 
2a10: 6f 77 6e 65 72 3a 20 53 29 20 2d 3e 20 52 65 73  owner: S) -> Res
2a20: 75 6c 74 3c 53 74 72 69 6e 67 3e 0a 09 77 68 65  ult<String>..whe
2a30: 72 65 20 53 3a 20 49 6e 74 6f 3c 69 36 34 3e 20  re S: Into<i64> 
2a40: 7b 0a 09 09 6c 65 74 20 6f 77 6e 65 72 20 3d 20  {...let owner = 
2a50: 6f 77 6e 65 72 2e 69 6e 74 6f 28 29 3b 0a 0a 09  owner.into();...
2a60: 09 6c 65 74 20 6d 75 74 20 72 65 70 6c 79 3a 20  .let mut reply: 
2a70: 56 65 63 3c 43 6f 77 3c 73 74 72 3e 3e 20 3d 20  Vec<Cow<str>> = 
2a80: 76 65 63 21 5b 5d 3b 0a 09 09 72 65 70 6c 79 2e  vec![];...reply.
2a90: 70 75 73 68 28 22 43 68 61 6e 6e 65 6c 73 3a 22  push("Channels:"
2aa0: 2e 69 6e 74 6f 28 29 29 3b 0a 09 09 6c 65 74 20  .into());...let 
2ab0: 72 6f 77 73 20 3d 20 73 71 6c 78 3a 3a 71 75 65  rows = sqlx::que
2ac0: 72 79 21 28 22 73 65 6c 65 63 74 20 73 6f 75 72  ry!("select sour
2ad0: 63 65 5f 69 64 2c 20 63 68 61 6e 6e 65 6c 2c 20  ce_id, channel, 
2ae0: 65 6e 61 62 6c 65 64 2c 20 75 72 6c 2c 20 69 76  enabled, url, iv
2af0: 5f 68 61 73 68 2c 20 75 72 6c 5f 72 65 20 66 72  _hash, url_re fr
2b00: 6f 6d 20 72 73 73 74 67 5f 73 6f 75 72 63 65 20  om rsstg_source 
2b10: 77 68 65 72 65 20 6f 77 6e 65 72 20 3d 20 24 31  where owner = $1
2b20: 20 6f 72 64 65 72 20 62 79 20 73 6f 75 72 63 65   order by source
2b30: 5f 69 64 22 2c 0a 09 09 09 6f 77 6e 65 72 29 2e  _id",....owner).
2b40: 66 65 74 63 68 5f 61 6c 6c 28 26 6d 75 74 20 2a  fetch_all(&mut *
2b50: 73 65 6c 66 2e 70 6f 6f 6c 2e 61 63 71 75 69 72  self.pool.acquir
2b60: 65 28 29 2e 61 77 61 69 74 3f 29 2e 61 77 61 69  e().await?).awai
2b70: 74 3f 3b 0a 09 09 66 6f 72 20 72 6f 77 20 69 6e  t?;...for row in
2b80: 20 72 6f 77 73 2e 69 74 65 72 28 29 20 7b 0a 09   rows.iter() {..
2b90: 09 09 72 65 70 6c 79 2e 70 75 73 68 28 66 6f 72  ..reply.push(for
2ba0: 6d 61 74 21 28 22 5c 6e 5c 5c 23 ef b8 8f e2 83  mat!("\n\\#ļøāƒ
2bb0: a3 20 7b 7d 20 5c 5c 2a ef b8 8f e2 83 a3 20 60  £ {} \\*ļøāƒ£ `
2bc0: 7b 7d 60 20 7b 7d 5c 6e f0 9f 94 97 20 60 7b 7d  {}` {}\nšŸ”— `{}
2bd0: 60 22 2c 20 72 6f 77 2e 73 6f 75 72 63 65 5f 69  `", row.source_i
2be0: 64 2c 20 72 6f 77 2e 63 68 61 6e 6e 65 6c 2c 0a  d, row.channel,.
2bf0: 09 09 09 09 6d 61 74 63 68 20 72 6f 77 2e 65 6e  ....match row.en
2c00: 61 62 6c 65 64 20 7b 0a 09 09 09 09 09 74 72 75  abled {......tru
2c10: 65 20 20 3d 3e 20 22 f0 9f 94 84 20 65 6e 61 62  e  => "šŸ”„ enab
2c20: 6c 65 64 22 2c 0a 09 09 09 09 09 66 61 6c 73 65  led",......false
2c30: 20 3d 3e 20 22 e2 9b 94 20 64 69 73 61 62 6c 65   => "ā›” disable
2c40: 64 22 2c 0a 09 09 09 09 7d 2c 20 72 6f 77 2e 75  d",.....}, row.u
2c50: 72 6c 29 2e 69 6e 74 6f 28 29 29 3b 0a 09 09 09  rl).into());....
2c60: 69 66 20 6c 65 74 20 53 6f 6d 65 28 68 61 73 68  if let Some(hash
2c70: 29 20 3d 20 26 72 6f 77 2e 69 76 5f 68 61 73 68  ) = &row.iv_hash
2c80: 20 7b 0a 09 09 09 09 72 65 70 6c 79 2e 70 75 73   {.....reply.pus
2c90: 68 28 66 6f 72 6d 61 74 21 28 22 49 56 3a 20 60  h(format!("IV: `
2ca0: 7b 7d 60 22 2c 20 68 61 73 68 29 2e 69 6e 74 6f  {}`", hash).into
2cb0: 28 29 29 3b 0a 09 09 09 7d 0a 09 09 09 69 66 20  ());....}....if 
2cc0: 6c 65 74 20 53 6f 6d 65 28 72 65 29 20 3d 20 26  let Some(re) = &
2cd0: 72 6f 77 2e 75 72 6c 5f 72 65 20 7b 0a 09 09 09  row.url_re {....
2ce0: 09 72 65 70 6c 79 2e 70 75 73 68 28 66 6f 72 6d  .reply.push(form
2cf0: 61 74 21 28 22 52 45 3a 20 60 7b 7d 60 22 2c 20  at!("RE: `{}`", 
2d00: 72 65 29 2e 69 6e 74 6f 28 29 29 3b 0a 09 09 09  re).into());....
2d10: 7d 0a 09 09 7d 3b 0a 09 09 4f 6b 28 72 65 70 6c  }...};...Ok(repl
2d20: 79 2e 6a 6f 69 6e 28 22 5c 6e 22 29 29 0a 09 7d  y.join("\n"))..}
2d30: 0a 7d 0a                                         .}.