Hex Artifact Content
Logged in as anonymous

Artifact 04985cd265d5d4781788eddda53c9405e281456995a30c270de2cd5a8640207c:


0000: 75 73 65 20 63 6f 6e 66 69 67 3b 0a 0a 75 73 65  use config;..use
0010: 20 74 6f 6b 69 6f 3b 0a 75 73 65 20 72 73 73 3b   tokio;.use rss;
0020: 0a 75 73 65 20 63 68 72 6f 6e 6f 3a 3a 44 61 74  .use chrono::Dat
0030: 65 54 69 6d 65 3b 0a 0a 75 73 65 20 72 65 67 65  eTime;..use rege
0040: 78 3a 3a 52 65 67 65 78 3b 0a 0a 75 73 65 20 74  x::Regex;..use t
0050: 6f 6b 69 6f 3a 3a 73 74 72 65 61 6d 3a 3a 53 74  okio::stream::St
0060: 72 65 61 6d 45 78 74 3b 0a 75 73 65 20 74 65 6c  reamExt;.use tel
0070: 65 67 72 61 6d 5f 62 6f 74 3a 3a 2a 3b 0a 0a 75  egram_bot::*;..u
0080: 73 65 20 73 71 6c 78 3a 3a 70 6f 73 74 67 72 65  se sqlx::postgre
0090: 73 3a 3a 50 67 50 6f 6f 6c 4f 70 74 69 6f 6e 73  s::PgPoolOptions
00a0: 3b 0a 75 73 65 20 73 71 6c 78 3a 3a 52 6f 77 3b  ;.use sqlx::Row;
00b0: 0a 0a 74 79 70 65 20 52 65 73 75 6c 74 3c 54 3e  ..type Result<T>
00c0: 20 3d 20 73 74 64 3a 3a 72 65 73 75 6c 74 3a 3a   = std::result::
00d0: 52 65 73 75 6c 74 3c 54 2c 20 42 6f 78 3c 64 79  Result<T, Box<dy
00e0: 6e 20 73 74 64 3a 3a 65 72 72 6f 72 3a 3a 45 72  n std::error::Er
00f0: 72 6f 72 3e 3e 3b 0a 0a 23 5b 64 65 72 69 76 65  ror>>;..#[derive
0100: 28 43 6c 6f 6e 65 29 5d 0a 73 74 72 75 63 74 20  (Clone)].struct 
0110: 43 6f 72 65 20 7b 0a 09 6f 77 6e 65 72 3a 20 69  Core {..owner: i
0120: 36 34 2c 0a 09 61 70 69 5f 6b 65 79 3a 20 53 74  64,..api_key: St
0130: 72 69 6e 67 2c 0a 09 6f 77 6e 65 72 5f 63 68 61  ring,..owner_cha
0140: 74 3a 20 55 73 65 72 49 64 2c 0a 09 74 67 3a 20  t: UserId,..tg: 
0150: 74 65 6c 65 67 72 61 6d 5f 62 6f 74 3a 3a 41 70  telegram_bot::Ap
0160: 69 2c 0a 09 6d 79 3a 20 55 73 65 72 2c 0a 09 70  i,..my: User,..p
0170: 6f 6f 6c 3a 20 73 71 6c 78 3a 3a 50 6f 6f 6c 3c  ool: sqlx::Pool<
0180: 73 71 6c 78 3a 3a 50 6f 73 74 67 72 65 73 3e 2c  sqlx::Postgres>,
0190: 0a 7d 0a 0a 69 6d 70 6c 20 43 6f 72 65 20 7b 0a  .}..impl Core {.
01a0: 09 61 73 79 6e 63 20 66 6e 20 6e 65 77 28 73 65  .async fn new(se
01b0: 74 74 69 6e 67 73 3a 20 63 6f 6e 66 69 67 3a 3a  ttings: config::
01c0: 43 6f 6e 66 69 67 29 20 2d 3e 20 52 65 73 75 6c  Config) -> Resul
01d0: 74 3c 43 6f 72 65 3e 20 7b 0a 09 09 6c 65 74 20  t<Core> {...let 
01e0: 6f 77 6e 65 72 20 3d 20 73 65 74 74 69 6e 67 73  owner = settings
01f0: 2e 67 65 74 5f 69 6e 74 28 22 6f 77 6e 65 72 22  .get_int("owner"
0200: 29 3f 3b 0a 09 09 6c 65 74 20 61 70 69 5f 6b 65  )?;...let api_ke
0210: 79 20 3d 20 73 65 74 74 69 6e 67 73 2e 67 65 74  y = settings.get
0220: 5f 73 74 72 28 22 61 70 69 5f 6b 65 79 22 29 3f  _str("api_key")?
0230: 3b 0a 09 09 6c 65 74 20 74 67 20 3d 20 41 70 69  ;...let tg = Api
0240: 3a 3a 6e 65 77 28 26 61 70 69 5f 6b 65 79 29 3b  ::new(&api_key);
0250: 0a 09 09 6c 65 74 20 63 6f 72 65 20 3d 20 43 6f  ...let core = Co
0260: 72 65 20 7b 0a 09 09 09 6f 77 6e 65 72 3a 20 6f  re {....owner: o
0270: 77 6e 65 72 2c 0a 09 09 09 61 70 69 5f 6b 65 79  wner,....api_key
0280: 3a 20 61 70 69 5f 6b 65 79 2e 63 6c 6f 6e 65 28  : api_key.clone(
0290: 29 2c 0a 09 09 09 6d 79 3a 20 74 67 2e 73 65 6e  ),....my: tg.sen
02a0: 64 28 74 65 6c 65 67 72 61 6d 5f 62 6f 74 3a 3a  d(telegram_bot::
02b0: 47 65 74 4d 65 29 2e 61 77 61 69 74 3f 2c 0a 09  GetMe).await?,..
02c0: 09 09 74 67 3a 20 74 67 2c 0a 09 09 09 6f 77 6e  ..tg: tg,....own
02d0: 65 72 5f 63 68 61 74 3a 20 55 73 65 72 49 64 3a  er_chat: UserId:
02e0: 3a 6e 65 77 28 6f 77 6e 65 72 29 2c 0a 09 09 09  :new(owner),....
02f0: 70 6f 6f 6c 3a 20 50 67 50 6f 6f 6c 4f 70 74 69  pool: PgPoolOpti
0300: 6f 6e 73 3a 3a 6e 65 77 28 29 2e 6d 61 78 5f 63  ons::new().max_c
0310: 6f 6e 6e 65 63 74 69 6f 6e 73 28 35 29 2e 63 6f  onnections(5).co
0320: 6e 6e 65 63 74 28 26 73 65 74 74 69 6e 67 73 2e  nnect(&settings.
0330: 67 65 74 5f 73 74 72 28 22 70 67 22 29 3f 29 2e  get_str("pg")?).
0340: 61 77 61 69 74 3f 2c 0a 09 09 7d 3b 0a 09 09 6c  await?,...};...l
0350: 65 74 20 63 6c 6f 6e 65 20 3d 20 63 6f 72 65 2e  et clone = core.
0360: 63 6c 6f 6e 65 28 29 3b 0a 09 09 74 6f 6b 69 6f  clone();...tokio
0370: 3a 3a 73 70 61 77 6e 28 61 73 79 6e 63 20 6d 6f  ::spawn(async mo
0380: 76 65 20 7b 0a 09 09 09 69 66 20 6c 65 74 20 45  ve {....if let E
0390: 72 72 28 65 72 72 29 20 3d 20 63 6c 6f 6e 65 2e  rr(err) = clone.
03a0: 61 75 74 6f 66 65 74 63 68 28 29 2e 61 77 61 69  autofetch().awai
03b0: 74 20 7b 0a 09 09 09 09 65 70 72 69 6e 74 6c 6e  t {.....eprintln
03c0: 21 28 22 63 6f 6e 6e 65 63 74 69 6f 6e 20 65 72  !("connection er
03d0: 72 6f 72 3a 20 7b 7d 22 2c 20 65 72 72 29 3b 0a  ror: {}", err);.
03e0: 09 09 09 7d 0a 09 09 7d 29 3b 0a 09 09 4f 6b 28  ...}...});...Ok(
03f0: 63 6f 72 65 29 0a 09 7d 0a 0a 09 66 6e 20 73 74  core)..}...fn st
0400: 72 65 61 6d 28 26 73 65 6c 66 29 20 2d 3e 20 74  ream(&self) -> t
0410: 65 6c 65 67 72 61 6d 5f 62 6f 74 3a 3a 55 70 64  elegram_bot::Upd
0420: 61 74 65 73 53 74 72 65 61 6d 20 7b 0a 09 09 73  atesStream {...s
0430: 65 6c 66 2e 74 67 2e 73 74 72 65 61 6d 28 29 0a  elf.tg.stream().
0440: 09 7d 0a 0a 09 66 6e 20 64 65 62 75 67 28 26 73  .}...fn debug(&s
0450: 65 6c 66 2c 20 6d 73 67 3a 20 26 73 74 72 29 20  elf, msg: &str) 
0460: 2d 3e 20 52 65 73 75 6c 74 3c 28 29 3e 20 7b 0a  -> Result<()> {.
0470: 09 09 73 65 6c 66 2e 74 67 2e 73 70 61 77 6e 28  ..self.tg.spawn(
0480: 53 65 6e 64 4d 65 73 73 61 67 65 3a 3a 6e 65 77  SendMessage::new
0490: 28 73 65 6c 66 2e 6f 77 6e 65 72 5f 63 68 61 74  (self.owner_chat
04a0: 2c 20 6d 73 67 29 29 3b 0a 09 09 4f 6b 28 28 29  , msg));...Ok(()
04b0: 29 0a 09 7d 0a 0a 09 61 73 79 6e 63 20 66 6e 20  )..}...async fn 
04c0: 63 68 65 63 6b 28 26 73 65 6c 66 2c 20 63 68 61  check(&self, cha
04d0: 6e 6e 65 6c 3a 20 26 73 74 72 2c 20 72 65 61 6c  nnel: &str, real
04e0: 3a 20 4f 70 74 69 6f 6e 3c 62 6f 6f 6c 3e 29 20  : Option<bool>) 
04f0: 2d 3e 20 52 65 73 75 6c 74 3c 28 29 3e 20 7b 0a  -> Result<()> {.
0500: 09 09 6d 61 74 63 68 20 73 71 6c 78 3a 3a 71 75  ..match sqlx::qu
0510: 65 72 79 28 22 73 65 6c 65 63 74 20 73 6f 75 72  ery("select sour
0520: 63 65 5f 69 64 2c 20 63 68 61 6e 6e 65 6c 5f 69  ce_id, channel_i
0530: 64 2c 20 75 72 6c 2c 20 6c 61 73 74 5f 66 65 74  d, url, last_fet
0540: 63 68 2c 20 69 76 5f 68 61 73 68 2c 20 6f 77 6e  ch, iv_hash, own
0550: 65 72 20 66 72 6f 6d 20 72 73 73 74 67 5f 73 6f  er from rsstg_so
0560: 75 72 63 65 20 6e 61 74 75 72 61 6c 20 6c 65 66  urce natural lef
0570: 74 20 6a 6f 69 6e 20 72 73 73 74 67 5f 63 68 61  t join rsstg_cha
0580: 6e 6e 65 6c 20 77 68 65 72 65 20 75 73 65 72 6e  nnel where usern
0590: 61 6d 65 20 3d 20 24 31 22 29 0a 09 09 09 2e 62  ame = $1").....b
05a0: 69 6e 64 28 63 68 61 6e 6e 65 6c 29 0a 09 09 09  ind(channel)....
05b0: 2e 66 65 74 63 68 5f 6f 6e 65 28 26 73 65 6c 66  .fetch_one(&self
05c0: 2e 70 6f 6f 6c 29 2e 61 77 61 69 74 20 7b 0a 09  .pool).await {..
05d0: 09 09 4f 6b 28 72 6f 77 29 20 3d 3e 20 7b 0a 09  ..Ok(row) => {..
05e0: 09 09 09 6c 65 74 20 69 64 3a 20 69 33 32 20 3d  ...let id: i32 =
05f0: 20 72 6f 77 2e 74 72 79 5f 67 65 74 28 22 73 6f   row.try_get("so
0600: 75 72 63 65 5f 69 64 22 29 3f 3b 0a 09 09 09 09  urce_id")?;.....
0610: 6c 65 74 20 63 68 61 6e 6e 65 6c 5f 69 64 3a 20  let channel_id: 
0620: 69 36 34 20 3d 20 72 6f 77 2e 74 72 79 5f 67 65  i64 = row.try_ge
0630: 74 28 22 63 68 61 6e 6e 65 6c 5f 69 64 22 29 3f  t("channel_id")?
0640: 3b 0a 09 09 09 09 6c 65 74 20 64 65 73 74 69 6e  ;.....let destin
0650: 61 74 69 6f 6e 20 3d 20 6d 61 74 63 68 20 72 65  ation = match re
0660: 61 6c 20 7b 0a 09 09 09 09 09 53 6f 6d 65 28 74  al {......Some(t
0670: 72 75 65 29 20 3d 3e 20 55 73 65 72 49 64 3a 3a  rue) => UserId::
0680: 6e 65 77 28 63 68 61 6e 6e 65 6c 5f 69 64 29 2c  new(channel_id),
0690: 0a 09 09 09 09 09 53 6f 6d 65 28 66 61 6c 73 65  ......Some(false
06a0: 29 20 7c 20 4e 6f 6e 65 20 3d 3e 20 55 73 65 72  ) | None => User
06b0: 49 64 3a 3a 6e 65 77 28 72 6f 77 2e 74 72 79 5f  Id::new(row.try_
06c0: 67 65 74 28 22 6f 77 6e 65 72 22 29 3f 29 2c 0a  get("owner")?),.
06d0: 09 09 09 09 7d 3b 0a 09 09 09 09 6c 65 74 20 75  ....};.....let u
06e0: 72 6c 3a 20 26 73 74 72 20 3d 20 72 6f 77 2e 74  rl: &str = row.t
06f0: 72 79 5f 67 65 74 28 22 75 72 6c 22 29 3f 3b 0a  ry_get("url")?;.
0700: 09 09 09 09 6c 65 74 20 6c 61 73 74 5f 66 65 74  ....let last_fet
0710: 63 68 3a 20 4f 70 74 69 6f 6e 3c 44 61 74 65 54  ch: Option<DateT
0720: 69 6d 65 3c 63 68 72 6f 6e 6f 3a 3a 46 69 78 65  ime<chrono::Fixe
0730: 64 4f 66 66 73 65 74 3e 3e 20 3d 20 72 6f 77 2e  dOffset>> = row.
0740: 74 72 79 5f 67 65 74 28 22 6c 61 73 74 5f 66 65  try_get("last_fe
0750: 74 63 68 22 29 3f 3b 0a 09 09 09 09 6c 65 74 20  tch")?;.....let 
0760: 6d 75 74 20 74 68 69 73 5f 66 65 74 63 68 3a 20  mut this_fetch: 
0770: 4f 70 74 69 6f 6e 3c 44 61 74 65 54 69 6d 65 3c  Option<DateTime<
0780: 63 68 72 6f 6e 6f 3a 3a 46 69 78 65 64 4f 66 66  chrono::FixedOff
0790: 73 65 74 3e 3e 20 3d 20 4e 6f 6e 65 3b 0a 09 09  set>> = None;...
07a0: 09 09 6c 65 74 20 69 76 5f 68 61 73 68 3a 20 4f  ..let iv_hash: O
07b0: 70 74 69 6f 6e 3c 26 73 74 72 3e 20 3d 20 72 6f  ption<&str> = ro
07c0: 77 2e 74 72 79 5f 67 65 74 28 22 69 76 5f 68 61  w.try_get("iv_ha
07d0: 73 68 22 29 3f 3b 0a 09 09 09 09 6d 61 74 63 68  sh")?;.....match
07e0: 20 72 73 73 3a 3a 43 68 61 6e 6e 65 6c 3a 3a 66   rss::Channel::f
07f0: 72 6f 6d 5f 75 72 6c 28 75 72 6c 29 20 7b 0a 09  rom_url(url) {..
0800: 09 09 09 09 4f 6b 28 66 65 65 64 29 20 3d 3e 20  ....Ok(feed) => 
0810: 7b 0a 09 09 09 09 09 09 73 65 6c 66 2e 64 65 62  {.......self.deb
0820: 75 67 28 26 66 6f 72 6d 61 74 21 28 22 23 20 74  ug(&format!("# t
0830: 69 74 6c 65 3a 7b 3a 3f 7d 20 74 74 6c 3a 7b 3a  itle:{:?} ttl:{:
0840: 3f 7d 20 68 6f 75 72 73 3a 7b 3a 3f 7d 20 64 61  ?} hours:{:?} da
0850: 79 73 3a 7b 3a 3f 7d 22 2c 20 66 65 65 64 2e 74  ys:{:?}", feed.t
0860: 69 74 6c 65 28 29 2c 20 66 65 65 64 2e 74 74 6c  itle(), feed.ttl
0870: 28 29 2c 20 66 65 65 64 2e 73 6b 69 70 5f 68 6f  (), feed.skip_ho
0880: 75 72 73 28 29 2c 20 66 65 65 64 2e 73 6b 69 70  urs(), feed.skip
0890: 5f 64 61 79 73 28 29 29 29 3f 3b 0a 09 09 09 09  _days()))?;.....
08a0: 09 09 66 6f 72 20 69 74 65 6d 20 69 6e 20 66 65  ..for item in fe
08b0: 65 64 2e 69 74 65 6d 73 28 29 20 7b 0a 09 09 09  ed.items() {....
08c0: 09 09 09 09 6c 65 74 20 64 61 74 65 20 3d 20 6d  ....let date = m
08d0: 61 74 63 68 20 69 74 65 6d 2e 70 75 62 5f 64 61  atch item.pub_da
08e0: 74 65 28 29 20 7b 0a 09 09 09 09 09 09 09 09 53  te() {.........S
08f0: 6f 6d 65 28 66 65 65 64 5f 64 61 74 65 29 20 3d  ome(feed_date) =
0900: 3e 20 44 61 74 65 54 69 6d 65 3a 3a 70 61 72 73  > DateTime::pars
0910: 65 5f 66 72 6f 6d 5f 72 66 63 32 38 32 32 28 66  e_from_rfc2822(f
0920: 65 65 64 5f 64 61 74 65 29 2c 0a 09 09 09 09 09  eed_date),......
0930: 09 09 09 4e 6f 6e 65 20 3d 3e 20 44 61 74 65 54  ...None => DateT
0940: 69 6d 65 3a 3a 70 61 72 73 65 5f 66 72 6f 6d 5f  ime::parse_from_
0950: 72 66 63 33 33 33 39 28 26 69 74 65 6d 2e 64 75  rfc3339(&item.du
0960: 62 6c 69 6e 5f 63 6f 72 65 5f 65 78 74 28 29 2e  blin_core_ext().
0970: 75 6e 77 72 61 70 28 29 2e 64 61 74 65 73 28 29  unwrap().dates()
0980: 5b 30 5d 29 2c 0a 09 09 09 09 09 09 09 7d 3f 3b  [0]),........}?;
0990: 0a 09 09 09 09 09 09 09 6c 65 74 20 75 72 6c 20  ........let url 
09a0: 3d 20 69 74 65 6d 2e 6c 69 6e 6b 28 29 2e 75 6e  = item.link().un
09b0: 77 72 61 70 28 29 2e 74 6f 5f 73 74 72 69 6e 67  wrap().to_string
09c0: 28 29 3b 0a 09 09 09 09 09 09 09 69 66 20 6c 61  ();........if la
09d0: 73 74 5f 66 65 74 63 68 20 3d 3d 20 4e 6f 6e 65  st_fetch == None
09e0: 20 7c 7c 20 64 61 74 65 20 3e 20 6c 61 73 74 5f   || date > last_
09f0: 66 65 74 63 68 2e 75 6e 77 72 61 70 28 29 20 7b  fetch.unwrap() {
0a00: 0a 09 09 09 09 09 09 09 09 6d 61 74 63 68 20 73  .........match s
0a10: 71 6c 78 3a 3a 71 75 65 72 79 28 22 73 65 6c 65  qlx::query("sele
0a20: 63 74 20 65 78 69 73 74 73 28 73 65 6c 65 63 74  ct exists(select
0a30: 20 74 72 75 65 20 66 72 6f 6d 20 72 73 73 74 67   true from rsstg
0a40: 5f 70 6f 73 74 20 77 68 65 72 65 20 75 72 6c 20  _post where url 
0a50: 3d 20 24 31 20 61 6e 64 20 73 6f 75 72 63 65 5f  = $1 and source_
0a60: 69 64 20 3d 20 24 32 29 20 61 73 20 65 78 69 73  id = $2) as exis
0a70: 74 73 3b 22 29 0a 09 09 09 09 09 09 09 09 09 2e  ts;")...........
0a80: 62 69 6e 64 28 26 75 72 6c 29 0a 09 09 09 09 09  bind(&url)......
0a90: 09 09 09 09 2e 62 69 6e 64 28 69 64 29 0a 09 09  .....bind(id)...
0aa0: 09 09 09 09 09 09 09 2e 66 65 74 63 68 5f 6f 6e  ........fetch_on
0ab0: 65 28 26 73 65 6c 66 2e 70 6f 6f 6c 29 2e 61 77  e(&self.pool).aw
0ac0: 61 69 74 20 7b 0a 09 09 09 09 09 09 09 09 09 4f  ait {..........O
0ad0: 6b 28 72 6f 77 29 20 3d 3e 20 7b 0a 09 09 09 09  k(row) => {.....
0ae0: 09 09 09 09 09 09 6c 65 74 20 65 78 69 73 74 73  ......let exists
0af0: 3a 20 62 6f 6f 6c 20 3d 20 72 6f 77 2e 74 72 79  : bool = row.try
0b00: 5f 67 65 74 28 22 65 78 69 73 74 73 22 29 3f 3b  _get("exists")?;
0b10: 0a 09 09 09 09 09 09 09 09 09 09 69 66 20 21 20  ...........if ! 
0b20: 65 78 69 73 74 73 20 7b 0a 09 09 09 09 09 09 09  exists {........
0b30: 09 09 09 09 69 66 20 74 68 69 73 5f 66 65 74 63  ....if this_fetc
0b40: 68 20 3d 3d 20 4e 6f 6e 65 20 7c 7c 20 64 61 74  h == None || dat
0b50: 65 20 3e 20 74 68 69 73 5f 66 65 74 63 68 2e 75  e > this_fetch.u
0b60: 6e 77 72 61 70 28 29 20 7b 0a 09 09 09 09 09 09  nwrap() {.......
0b70: 09 09 09 09 09 09 74 68 69 73 5f 66 65 74 63 68  ......this_fetch
0b80: 20 3d 20 53 6f 6d 65 28 64 61 74 65 29 3b 0a 09   = Some(date);..
0b90: 09 09 09 09 09 09 09 09 09 09 7d 0a 09 09 09 09  ..........}.....
0ba0: 09 09 09 09 09 09 09 6d 61 74 63 68 20 73 65 6c  .......match sel
0bb0: 66 2e 74 67 2e 73 65 6e 64 28 20 6d 61 74 63 68  f.tg.send( match
0bc0: 20 69 76 5f 68 61 73 68 20 7b 0a 09 09 09 09 09   iv_hash {......
0bd0: 09 09 09 09 09 09 09 09 53 6f 6d 65 28 78 29 20  ........Some(x) 
0be0: 3d 3e 20 53 65 6e 64 4d 65 73 73 61 67 65 3a 3a  => SendMessage::
0bf0: 6e 65 77 28 64 65 73 74 69 6e 61 74 69 6f 6e 2c  new(destination,
0c00: 20 66 6f 72 6d 61 74 21 28 22 3c 61 20 68 72 65   format!("<a hre
0c10: 66 3d 5c 22 68 74 74 70 73 3a 2f 2f 74 2e 6d 65  f=\"https://t.me
0c20: 2f 69 76 3f 75 72 6c 3d 7b 7d 26 72 68 61 73 68  /iv?url={}&rhash
0c30: 3d 7b 7d 5c 22 3e 20 3c 2f 61 3e 7b 30 7d 22 2c  ={}\"> </a>{0}",
0c40: 20 75 72 6c 2c 20 78 29 29 2c 0a 09 09 09 09 09   url, x)),......
0c50: 09 09 09 09 09 09 09 09 4e 6f 6e 65 20 3d 3e 20  ........None => 
0c60: 53 65 6e 64 4d 65 73 73 61 67 65 3a 3a 6e 65 77  SendMessage::new
0c70: 28 64 65 73 74 69 6e 61 74 69 6f 6e 2c 20 66 6f  (destination, fo
0c80: 72 6d 61 74 21 28 22 7b 7d 22 2c 20 75 72 6c 29  rmat!("{}", url)
0c90: 29 2c 0a 09 09 09 09 09 09 09 09 09 09 09 09 7d  ),.............}
0ca0: 2e 70 61 72 73 65 5f 6d 6f 64 65 28 74 79 70 65  .parse_mode(type
0cb0: 73 3a 3a 50 61 72 73 65 4d 6f 64 65 3a 3a 48 74  s::ParseMode::Ht
0cc0: 6d 6c 29 29 2e 61 77 61 69 74 20 7b 0a 09 09 09  ml)).await {....
0cd0: 09 09 09 09 09 09 09 09 09 4f 6b 28 5f 29 20 3d  .........Ok(_) =
0ce0: 3e 20 7b 0a 09 09 09 09 09 09 09 09 09 09 09 09  > {.............
0cf0: 09 6d 61 74 63 68 20 73 71 6c 78 3a 3a 71 75 65  .match sqlx::que
0d00: 72 79 28 22 69 6e 73 65 72 74 20 69 6e 74 6f 20  ry("insert into 
0d10: 72 73 73 74 67 5f 70 6f 73 74 20 28 73 6f 75 72  rsstg_post (sour
0d20: 63 65 5f 69 64 2c 20 70 6f 73 74 65 64 2c 20 75  ce_id, posted, u
0d30: 72 6c 29 20 76 61 6c 75 65 73 20 28 24 31 2c 20  rl) values ($1, 
0d40: 24 32 2c 20 24 33 29 3b 22 29 0a 09 09 09 09 09  $2, $3);")......
0d50: 09 09 09 09 09 09 09 09 09 2e 62 69 6e 64 28 69  ..........bind(i
0d60: 64 29 0a 09 09 09 09 09 09 09 09 09 09 09 09 09  d)..............
0d70: 09 2e 62 69 6e 64 28 64 61 74 65 29 0a 09 09 09  ..bind(date)....
0d80: 09 09 09 09 09 09 09 09 09 09 09 2e 62 69 6e 64  ............bind
0d90: 28 75 72 6c 29 0a 09 09 09 09 09 09 09 09 09 09  (url)...........
0da0: 09 09 09 09 2e 65 78 65 63 75 74 65 28 26 73 65  .....execute(&se
0db0: 6c 66 2e 70 6f 6f 6c 29 2e 61 77 61 69 74 20 7b  lf.pool).await {
0dc0: 0a 09 09 09 09 09 09 09 09 09 09 09 09 09 09 09  ................
0dd0: 4f 6b 28 5f 29 20 3d 3e 20 7b 7d 2c 0a 09 09 09  Ok(_) => {},....
0de0: 09 09 09 09 09 09 09 09 09 09 09 09 45 72 72 28  ............Err(
0df0: 65 72 72 29 20 3d 3e 20 7b 0a 09 09 09 09 09 09  err) => {.......
0e00: 09 09 09 09 09 09 09 09 09 09 73 65 6c 66 2e 64  ..........self.d
0e10: 65 62 75 67 28 26 65 72 72 2e 74 6f 5f 73 74 72  ebug(&err.to_str
0e20: 69 6e 67 28 29 29 3f 3b 0a 09 09 09 09 09 09 09  ing())?;........
0e30: 09 09 09 09 09 09 09 09 7d 2c 0a 09 09 09 09 09  ........},......
0e40: 09 09 09 09 09 09 09 09 7d 3b 0a 09 09 09 09 09  ........};......
0e50: 09 09 09 09 09 09 09 7d 2c 0a 09 09 09 09 09 09  .......},.......
0e60: 09 09 09 09 09 09 45 72 72 28 65 72 72 29 20 3d  ......Err(err) =
0e70: 3e 20 7b 0a 09 09 09 09 09 09 09 09 09 09 09 09  > {.............
0e80: 09 73 65 6c 66 2e 64 65 62 75 67 28 26 65 72 72  .self.debug(&err
0e90: 2e 74 6f 5f 73 74 72 69 6e 67 28 29 29 3f 3b 0a  .to_string())?;.
0ea0: 09 09 09 09 09 09 09 09 09 09 09 09 7d 2c 0a 09  ............},..
0eb0: 09 09 09 09 09 09 09 09 09 09 7d 3b 0a 09 09 09  ..........};....
0ec0: 09 09 09 09 09 09 09 09 74 6f 6b 69 6f 3a 3a 74  ........tokio::t
0ed0: 69 6d 65 3a 3a 64 65 6c 61 79 5f 66 6f 72 28 73  ime::delay_for(s
0ee0: 74 64 3a 3a 74 69 6d 65 3a 3a 44 75 72 61 74 69  td::time::Durati
0ef0: 6f 6e 3a 3a 6e 65 77 28 34 2c 20 30 29 29 2e 61  on::new(4, 0)).a
0f00: 77 61 69 74 3b 0a 09 09 09 09 09 09 09 09 09 09  wait;...........
0f10: 7d 0a 09 09 09 09 09 09 09 09 09 7d 2c 0a 09 09  }..........},...
0f20: 09 09 09 09 09 09 09 45 72 72 28 65 72 72 29 20  .......Err(err) 
0f30: 3d 3e 20 7b 0a 09 09 09 09 09 09 09 09 09 09 73  => {...........s
0f40: 65 6c 66 2e 64 65 62 75 67 28 26 65 72 72 2e 74  elf.debug(&err.t
0f50: 6f 5f 73 74 72 69 6e 67 28 29 29 3f 3b 0a 09 09  o_string())?;...
0f60: 09 09 09 09 09 09 09 7d 2c 0a 09 09 09 09 09 09  .......},.......
0f70: 09 09 7d 3b 0a 09 09 09 09 09 09 09 7d 3b 0a 09  ..};........};..
0f80: 09 09 09 09 09 7d 3b 0a 09 09 09 09 09 09 2f 2f  .....};.......//
0f90: 20 75 70 64 61 74 65 20 6c 61 73 74 5f 66 65 74   update last_fet
0fa0: 63 68 0a 09 09 09 09 09 09 69 66 20 74 68 69 73  ch.......if this
0fb0: 5f 66 65 74 63 68 20 21 3d 20 4e 6f 6e 65 20 26  _fetch != None &
0fc0: 26 20 28 6c 61 73 74 5f 66 65 74 63 68 20 3d 3d  & (last_fetch ==
0fd0: 20 4e 6f 6e 65 20 7c 7c 20 74 68 69 73 5f 66 65   None || this_fe
0fe0: 74 63 68 2e 75 6e 77 72 61 70 28 29 20 3e 20 6c  tch.unwrap() > l
0ff0: 61 73 74 5f 66 65 74 63 68 2e 75 6e 77 72 61 70  ast_fetch.unwrap
1000: 28 29 29 20 7b 0a 09 09 09 09 09 09 09 6d 61 74  ()) {........mat
1010: 63 68 20 73 71 6c 78 3a 3a 71 75 65 72 79 28 22  ch sqlx::query("
1020: 75 70 64 61 74 65 20 72 73 73 74 67 5f 73 6f 75  update rsstg_sou
1030: 72 63 65 20 73 65 74 20 6c 61 73 74 5f 66 65 74  rce set last_fet
1040: 63 68 20 3d 20 63 61 73 65 20 77 68 65 6e 20 28  ch = case when (
1050: 6c 61 73 74 5f 66 65 74 63 68 20 3c 20 24 31 29  last_fetch < $1)
1060: 20 74 68 65 6e 20 24 31 20 65 6c 73 65 20 6c 61   then $1 else la
1070: 73 74 5f 66 65 74 63 68 20 65 6e 64 20 77 68 65  st_fetch end whe
1080: 72 65 20 73 6f 75 72 63 65 5f 69 64 20 3d 20 24  re source_id = $
1090: 32 3b 22 29 0a 09 09 09 09 09 09 09 09 2e 62 69  2;")..........bi
10a0: 6e 64 28 74 68 69 73 5f 66 65 74 63 68 2e 75 6e  nd(this_fetch.un
10b0: 77 72 61 70 28 29 29 0a 09 09 09 09 09 09 09 09  wrap()).........
10c0: 2e 62 69 6e 64 28 69 64 29 0a 09 09 09 09 09 09  .bind(id).......
10d0: 09 09 2e 65 78 65 63 75 74 65 28 26 73 65 6c 66  ...execute(&self
10e0: 2e 70 6f 6f 6c 29 2e 61 77 61 69 74 20 7b 0a 09  .pool).await {..
10f0: 09 09 09 09 09 09 09 4f 6b 28 5f 29 20 3d 3e 20  .......Ok(_) => 
1100: 7b 7d 2c 0a 09 09 09 09 09 09 09 09 45 72 72 28  {},.........Err(
1110: 65 72 72 29 20 3d 3e 20 7b 0a 09 09 09 09 09 09  err) => {.......
1120: 09 09 09 73 65 6c 66 2e 64 65 62 75 67 28 26 65  ...self.debug(&e
1130: 72 72 2e 74 6f 5f 73 74 72 69 6e 67 28 29 29 3f  rr.to_string())?
1140: 3b 0a 09 09 09 09 09 09 09 09 7d 2c 0a 09 09 09  ;.........},....
1150: 09 09 09 09 7d 3b 0a 09 09 09 09 09 09 7d 0a 09  ....};.......}..
1160: 09 09 09 09 7d 2c 0a 09 09 09 09 09 45 72 72 28  ....},......Err(
1170: 65 72 72 29 20 3d 3e 20 7b 0a 09 09 09 09 09 09  err) => {.......
1180: 73 65 6c 66 2e 64 65 62 75 67 28 26 65 72 72 2e  self.debug(&err.
1190: 74 6f 5f 73 74 72 69 6e 67 28 29 29 3f 3b 0a 09  to_string())?;..
11a0: 09 09 09 09 7d 2c 0a 09 09 09 09 7d 3b 0a 09 09  ....},.....};...
11b0: 09 09 6d 61 74 63 68 20 73 71 6c 78 3a 3a 71 75  ..match sqlx::qu
11c0: 65 72 79 28 22 75 70 64 61 74 65 20 72 73 73 74  ery("update rsst
11d0: 67 5f 73 6f 75 72 63 65 20 73 65 74 20 6c 61 73  g_source set las
11e0: 74 5f 73 63 72 61 70 65 20 3d 20 6e 6f 77 28 29  t_scrape = now()
11f0: 20 77 68 65 72 65 20 73 6f 75 72 63 65 5f 69 64   where source_id
1200: 20 3d 20 24 31 3b 22 29 0a 09 09 09 09 09 2e 62   = $1;").......b
1210: 69 6e 64 28 69 64 29 0a 09 09 09 09 09 2e 65 78  ind(id).......ex
1220: 65 63 75 74 65 28 26 73 65 6c 66 2e 70 6f 6f 6c  ecute(&self.pool
1230: 29 2e 61 77 61 69 74 20 7b 0a 09 09 09 09 09 4f  ).await {......O
1240: 6b 28 5f 29 20 3d 3e 20 7b 7d 2c 0a 09 09 09 09  k(_) => {},.....
1250: 09 45 72 72 28 65 72 72 29 20 3d 3e 20 7b 0a 09  .Err(err) => {..
1260: 09 09 09 09 09 73 65 6c 66 2e 64 65 62 75 67 28  .....self.debug(
1270: 26 65 72 72 2e 74 6f 5f 73 74 72 69 6e 67 28 29  &err.to_string()
1280: 29 3f 3b 0a 09 09 09 09 09 7d 2c 0a 09 09 09 09  )?;......},.....
1290: 7d 3b 0a 09 09 09 7d 2c 0a 09 09 09 45 72 72 28  };....},....Err(
12a0: 65 72 72 29 20 3d 3e 20 7b 0a 09 09 09 09 73 65  err) => {.....se
12b0: 6c 66 2e 64 65 62 75 67 28 26 65 72 72 2e 74 6f  lf.debug(&err.to
12c0: 5f 73 74 72 69 6e 67 28 29 29 3f 3b 0a 09 09 09  _string())?;....
12d0: 7d 2c 0a 09 09 7d 3b 0a 09 09 4f 6b 28 28 29 29  },...};...Ok(())
12e0: 0a 09 7d 0a 0a 09 61 73 79 6e 63 20 66 6e 20 63  ..}...async fn c
12f0: 6c 65 61 6e 28 26 73 65 6c 66 2c 20 73 6f 75 72  lean(&self, sour
1300: 63 65 5f 69 64 3a 20 69 33 32 29 20 2d 3e 20 52  ce_id: i32) -> R
1310: 65 73 75 6c 74 3c 28 29 3e 20 7b 0a 09 09 66 6f  esult<()> {...fo
1320: 72 20 71 75 65 72 79 20 69 6e 20 76 65 63 21 5b  r query in vec![
1330: 22 64 65 6c 65 74 65 20 66 72 6f 6d 20 72 73 73  "delete from rss
1340: 74 67 5f 70 6f 73 74 20 77 68 65 72 65 20 73 6f  tg_post where so
1350: 75 72 63 65 5f 69 64 20 3d 20 24 31 3b 22 2c 20  urce_id = $1;", 
1360: 22 75 70 64 61 74 65 20 72 73 73 74 67 5f 73 6f  "update rsstg_so
1370: 75 72 63 65 20 73 65 74 20 6c 61 73 74 5f 66 65  urce set last_fe
1380: 74 63 68 20 3d 20 4e 55 4c 4c 20 77 68 65 72 65  tch = NULL where
1390: 20 73 6f 75 72 63 65 5f 69 64 20 3d 20 24 31 3b   source_id = $1;
13a0: 22 5d 20 7b 0a 09 09 09 6d 61 74 63 68 20 73 71  "] {....match sq
13b0: 6c 78 3a 3a 71 75 65 72 79 28 71 75 65 72 79 29  lx::query(query)
13c0: 0a 09 09 09 09 2e 62 69 6e 64 28 73 6f 75 72 63  ......bind(sourc
13d0: 65 5f 69 64 29 0a 09 09 09 09 2e 65 78 65 63 75  e_id)......execu
13e0: 74 65 28 26 73 65 6c 66 2e 70 6f 6f 6c 29 2e 61  te(&self.pool).a
13f0: 77 61 69 74 20 7b 0a 09 09 09 09 4f 6b 28 5f 29  wait {.....Ok(_)
1400: 20 3d 3e 20 7b 7d 2c 0a 09 09 09 09 45 72 72 28   => {},.....Err(
1410: 65 72 72 29 20 3d 3e 20 7b 0a 09 09 09 09 09 73  err) => {......s
1420: 65 6c 66 2e 64 65 62 75 67 28 26 65 72 72 2e 74  elf.debug(&err.t
1430: 6f 5f 73 74 72 69 6e 67 28 29 29 3f 3b 0a 09 09  o_string())?;...
1440: 09 09 7d 2c 0a 09 09 09 7d 0a 09 09 7d 0a 09 09  ..},....}...}...
1450: 4f 6b 28 28 29 29 0a 09 7d 0a 0a 09 61 73 79 6e  Ok(())..}...asyn
1460: 63 20 66 6e 20 65 6e 61 62 6c 65 28 26 73 65 6c  c fn enable(&sel
1470: 66 2c 20 75 73 65 72 3a 20 55 73 65 72 49 64 2c  f, user: UserId,
1480: 20 63 68 61 6e 6e 65 6c 3a 20 26 73 74 72 29 20   channel: &str) 
1490: 2d 3e 20 52 65 73 75 6c 74 3c 28 29 3e 20 7b 0a  -> Result<()> {.
14a0: 09 09 6d 61 74 63 68 20 73 71 6c 78 3a 3a 71 75  ..match sqlx::qu
14b0: 65 72 79 28 22 75 70 64 61 74 65 20 72 73 73 74  ery("update rsst
14c0: 67 5f 73 6f 75 72 63 65 20 73 65 74 20 65 6e 61  g_source set ena
14d0: 62 6c 65 64 20 3d 20 74 72 75 65 20 66 72 6f 6d  bled = true from
14e0: 20 72 73 73 74 67 5f 63 68 61 6e 6e 65 6c 20 77   rsstg_channel w
14f0: 68 65 72 65 20 72 73 73 74 67 5f 63 68 61 6e 6e  here rsstg_chann
1500: 65 6c 2e 63 68 61 6e 6e 65 6c 5f 69 64 20 3d 20  el.channel_id = 
1510: 72 73 73 74 67 5f 73 6f 75 72 63 65 2e 63 68 61  rsstg_source.cha
1520: 6e 6e 65 6c 5f 69 64 20 61 6e 64 20 72 73 73 74  nnel_id and rsst
1530: 67 5f 63 68 61 6e 6e 65 6c 2e 75 73 65 72 6e 61  g_channel.userna
1540: 6d 65 20 3d 20 24 31 20 61 6e 64 20 6f 77 6e 65  me = $1 and owne
1550: 72 20 3d 20 24 32 22 29 0a 09 09 09 2e 62 69 6e  r = $2").....bin
1560: 64 28 63 68 61 6e 6e 65 6c 29 0a 09 09 09 2e 62  d(channel).....b
1570: 69 6e 64 28 69 36 34 3a 3a 66 72 6f 6d 28 75 73  ind(i64::from(us
1580: 65 72 29 29 0a 09 09 09 2e 65 78 65 63 75 74 65  er)).....execute
1590: 28 26 73 65 6c 66 2e 70 6f 6f 6c 29 2e 61 77 61  (&self.pool).awa
15a0: 69 74 20 7b 0a 09 09 09 4f 6b 28 5f 29 20 3d 3e  it {....Ok(_) =>
15b0: 20 7b 7d 2c 0a 09 09 09 45 72 72 28 65 72 72 29   {},....Err(err)
15c0: 20 3d 3e 20 7b 0a 09 09 09 09 73 65 6c 66 2e 64   => {.....self.d
15d0: 65 62 75 67 28 26 65 72 72 2e 74 6f 5f 73 74 72  ebug(&err.to_str
15e0: 69 6e 67 28 29 29 3f 3b 0a 09 09 09 7d 2c 0a 09  ing())?;....},..
15f0: 09 7d 0a 09 09 4f 6b 28 28 29 29 0a 09 7d 0a 0a  .}...Ok(())..}..
1600: 09 61 73 79 6e 63 20 66 6e 20 64 69 73 61 62 6c  .async fn disabl
1610: 65 28 26 73 65 6c 66 2c 20 75 73 65 72 3a 20 55  e(&self, user: U
1620: 73 65 72 49 64 2c 20 63 68 61 6e 6e 65 6c 3a 20  serId, channel: 
1630: 26 73 74 72 29 20 2d 3e 20 52 65 73 75 6c 74 3c  &str) -> Result<
1640: 28 29 3e 20 7b 0a 09 09 6d 61 74 63 68 20 73 71  ()> {...match sq
1650: 6c 78 3a 3a 71 75 65 72 79 28 22 75 70 64 61 74  lx::query("updat
1660: 65 20 72 73 73 74 67 5f 73 6f 75 72 63 65 20 73  e rsstg_source s
1670: 65 74 20 65 6e 61 62 6c 65 64 20 3d 20 66 61 6c  et enabled = fal
1680: 73 65 20 66 72 6f 6d 20 72 73 73 74 67 5f 63 68  se from rsstg_ch
1690: 61 6e 6e 65 6c 20 77 68 65 72 65 20 72 73 73 74  annel where rsst
16a0: 67 5f 63 68 61 6e 6e 65 6c 2e 63 68 61 6e 6e 65  g_channel.channe
16b0: 6c 5f 69 64 20 3d 20 72 73 73 74 67 5f 73 6f 75  l_id = rsstg_sou
16c0: 72 63 65 2e 63 68 61 6e 6e 65 6c 5f 69 64 20 61  rce.channel_id a
16d0: 6e 64 20 72 73 73 74 67 5f 63 68 61 6e 6e 65 6c  nd rsstg_channel
16e0: 2e 75 73 65 72 6e 61 6d 65 20 3d 20 24 31 20 61  .username = $1 a
16f0: 6e 64 20 6f 77 6e 65 72 20 3d 20 24 32 22 29 0a  nd owner = $2").
1700: 09 09 09 2e 62 69 6e 64 28 63 68 61 6e 6e 65 6c  ....bind(channel
1710: 29 0a 09 09 09 2e 62 69 6e 64 28 69 36 34 3a 3a  ).....bind(i64::
1720: 66 72 6f 6d 28 75 73 65 72 29 29 0a 09 09 09 2e  from(user)).....
1730: 65 78 65 63 75 74 65 28 26 73 65 6c 66 2e 70 6f  execute(&self.po
1740: 6f 6c 29 2e 61 77 61 69 74 20 7b 0a 09 09 09 4f  ol).await {....O
1750: 6b 28 5f 29 20 3d 3e 20 7b 7d 2c 0a 09 09 09 45  k(_) => {},....E
1760: 72 72 28 65 72 72 29 20 3d 3e 20 7b 0a 09 09 09  rr(err) => {....
1770: 09 73 65 6c 66 2e 64 65 62 75 67 28 26 65 72 72  .self.debug(&err
1780: 2e 74 6f 5f 73 74 72 69 6e 67 28 29 29 3f 3b 0a  .to_string())?;.
1790: 09 09 09 7d 2c 0a 09 09 7d 0a 09 09 4f 6b 28 28  ...},...}...Ok((
17a0: 29 29 0a 09 7d 0a 0a 09 61 73 79 6e 63 20 66 6e  ))..}...async fn
17b0: 20 61 75 74 6f 66 65 74 63 68 28 26 73 65 6c 66   autofetch(&self
17c0: 29 20 2d 3e 20 52 65 73 75 6c 74 3c 28 29 3e 20  ) -> Result<()> 
17d0: 7b 0a 09 09 6c 65 74 20 6d 75 74 20 64 65 6c 61  {...let mut dela
17e0: 79 20 3d 20 63 68 72 6f 6e 6f 3a 3a 44 75 72 61  y = chrono::Dura
17f0: 74 69 6f 6e 3a 3a 6d 69 6e 75 74 65 73 28 35 29  tion::minutes(5)
1800: 3b 0a 09 09 6c 65 74 20 6d 75 74 20 6e 65 78 74  ;...let mut next
1810: 5f 66 65 74 63 68 3a 20 44 61 74 65 54 69 6d 65  _fetch: DateTime
1820: 3c 63 68 72 6f 6e 6f 3a 3a 4c 6f 63 61 6c 3e 3b  <chrono::Local>;
1830: 0a 09 09 6c 65 74 20 6d 75 74 20 6e 6f 77 3b 0a  ...let mut now;.
1840: 09 09 6c 6f 6f 70 20 7b 0a 09 09 09 6c 65 74 20  ..loop {....let 
1850: 6d 75 74 20 72 6f 77 73 20 3d 20 73 71 6c 78 3a  mut rows = sqlx:
1860: 3a 71 75 65 72 79 28 22 73 65 6c 65 63 74 20 73  :query("select s
1870: 6f 75 72 63 65 5f 69 64 2c 20 75 73 65 72 6e 61  ource_id, userna
1880: 6d 65 2c 20 6e 65 78 74 5f 66 65 74 63 68 20 66  me, next_fetch f
1890: 72 6f 6d 20 72 73 73 74 67 5f 6f 72 64 65 72 20  rom rsstg_order 
18a0: 6e 61 74 75 72 61 6c 20 6c 65 66 74 20 6a 6f 69  natural left joi
18b0: 6e 20 72 73 73 74 67 5f 73 6f 75 72 63 65 20 6e  n rsstg_source n
18c0: 61 74 75 72 61 6c 20 6c 65 66 74 20 6a 6f 69 6e  atural left join
18d0: 20 72 73 73 74 67 5f 63 68 61 6e 6e 65 6c 3b 22   rsstg_channel;"
18e0: 29 0a 09 09 09 09 2e 66 65 74 63 68 28 26 73 65  )......fetch(&se
18f0: 6c 66 2e 70 6f 6f 6c 29 3b 0a 09 09 09 77 68 69  lf.pool);....whi
1900: 6c 65 20 6c 65 74 20 53 6f 6d 65 28 72 6f 77 29  le let Some(row)
1910: 20 3d 20 72 6f 77 73 2e 74 72 79 5f 6e 65 78 74   = rows.try_next
1920: 28 29 2e 61 77 61 69 74 2e 75 6e 77 72 61 70 28  ().await.unwrap(
1930: 29 20 7b 0a 09 09 09 09 6e 6f 77 20 3d 20 63 68  ) {.....now = ch
1940: 72 6f 6e 6f 3a 3a 4c 6f 63 61 6c 3a 3a 6e 6f 77  rono::Local::now
1950: 28 29 3b 0a 09 09 09 09 6c 65 74 20 73 6f 75 72  ();.....let sour
1960: 63 65 5f 69 64 3a 20 69 33 32 20 3d 20 72 6f 77  ce_id: i32 = row
1970: 2e 74 72 79 5f 67 65 74 28 22 73 6f 75 72 63 65  .try_get("source
1980: 5f 69 64 22 29 3f 3b 0a 09 09 09 09 6e 65 78 74  _id")?;.....next
1990: 5f 66 65 74 63 68 20 3d 20 72 6f 77 2e 74 72 79  _fetch = row.try
19a0: 5f 67 65 74 28 22 6e 65 78 74 5f 66 65 74 63 68  _get("next_fetch
19b0: 22 29 3f 3b 0a 09 09 09 09 69 66 20 6e 65 78 74  ")?;.....if next
19c0: 5f 66 65 74 63 68 20 3c 20 6e 6f 77 20 7b 0a 09  _fetch < now {..
19d0: 09 09 09 09 6d 61 74 63 68 20 73 71 6c 78 3a 3a  ....match sqlx::
19e0: 71 75 65 72 79 28 22 75 70 64 61 74 65 20 72 73  query("update rs
19f0: 73 74 67 5f 73 6f 75 72 63 65 20 73 65 74 20 6c  stg_source set l
1a00: 61 73 74 5f 73 63 72 61 70 65 20 3d 20 6e 6f 77  ast_scrape = now
1a10: 28 29 20 2b 20 69 6e 74 65 72 76 61 6c 20 27 31  () + interval '1
1a20: 20 68 6f 75 72 27 20 77 68 65 72 65 20 73 6f 75   hour' where sou
1a30: 72 63 65 5f 69 64 20 3d 20 24 31 3b 22 29 0a 09  rce_id = $1;")..
1a40: 09 09 09 09 09 2e 62 69 6e 64 28 73 6f 75 72 63  ......bind(sourc
1a50: 65 5f 69 64 29 0a 09 09 09 09 09 09 2e 65 78 65  e_id)........exe
1a60: 63 75 74 65 28 26 73 65 6c 66 2e 70 6f 6f 6c 29  cute(&self.pool)
1a70: 2e 61 77 61 69 74 20 7b 0a 09 09 09 09 09 09 4f  .await {.......O
1a80: 6b 28 5f 29 20 3d 3e 20 7b 7d 2c 0a 09 09 09 09  k(_) => {},.....
1a90: 09 09 45 72 72 28 65 72 72 29 20 3d 3e 20 7b 0a  ..Err(err) => {.
1aa0: 09 09 09 09 09 09 09 73 65 6c 66 2e 64 65 62 75  .......self.debu
1ab0: 67 28 26 65 72 72 2e 74 6f 5f 73 74 72 69 6e 67  g(&err.to_string
1ac0: 28 29 29 3f 3b 0a 09 09 09 09 09 09 7d 2c 0a 09  ())?;.......},..
1ad0: 09 09 09 09 7d 3b 0a 09 09 09 09 09 6c 65 74 20  ....};......let 
1ae0: 63 6c 6f 6e 65 20 3d 20 73 65 6c 66 2e 63 6c 6f  clone = self.clo
1af0: 6e 65 28 29 3b 0a 09 09 09 09 09 6c 65 74 20 75  ne();......let u
1b00: 73 65 72 6e 61 6d 65 3a 20 53 74 72 69 6e 67 20  sername: String 
1b10: 3d 20 72 6f 77 2e 74 72 79 5f 67 65 74 28 22 75  = row.try_get("u
1b20: 73 65 72 6e 61 6d 65 22 29 3f 3b 0a 09 09 09 09  sername")?;.....
1b30: 09 6c 65 74 20 75 73 65 72 6e 61 6d 65 20 3d 20  .let username = 
1b40: 75 73 65 72 6e 61 6d 65 2e 63 6c 6f 6e 65 28 29  username.clone()
1b50: 3b 0a 09 09 09 09 09 74 6f 6b 69 6f 3a 3a 73 70  ;......tokio::sp
1b60: 61 77 6e 28 61 73 79 6e 63 20 6d 6f 76 65 20 7b  awn(async move {
1b70: 0a 09 09 09 09 09 09 69 66 20 6c 65 74 20 45 72  .......if let Er
1b80: 72 28 65 72 72 29 20 3d 20 63 6c 6f 6e 65 2e 63  r(err) = clone.c
1b90: 68 65 63 6b 28 26 75 73 65 72 6e 61 6d 65 2c 20  heck(&username, 
1ba0: 53 6f 6d 65 28 74 72 75 65 29 29 2e 61 77 61 69  Some(true)).awai
1bb0: 74 20 7b 0a 09 09 09 09 09 09 09 65 70 72 69 6e  t {........eprin
1bc0: 74 6c 6e 21 28 22 63 6f 6e 6e 65 63 74 69 6f 6e  tln!("connection
1bd0: 20 65 72 72 6f 72 3a 20 7b 7d 22 2c 20 65 72 72   error: {}", err
1be0: 29 3b 0a 09 09 09 09 09 09 7d 0a 09 09 09 09 09  );.......}......
1bf0: 7d 29 3b 0a 09 09 09 09 09 2f 2f 26 73 65 6c 66  });......//&self
1c00: 2e 63 68 65 63 6b 28 72 6f 77 2e 74 72 79 5f 67  .check(row.try_g
1c10: 65 74 28 22 75 73 65 72 6e 61 6d 65 22 29 3f 2c  et("username")?,
1c20: 20 53 6f 6d 65 28 74 72 75 65 29 29 2e 61 77 61   Some(true)).awa
1c30: 69 74 3f 3b 0a 09 09 09 09 7d 20 65 6c 73 65 20  it?;.....} else 
1c40: 7b 0a 09 09 09 09 09 69 66 20 6e 65 78 74 5f 66  {......if next_f
1c50: 65 74 63 68 20 2d 20 6e 6f 77 20 3c 20 64 65 6c  etch - now < del
1c60: 61 79 20 7b 0a 09 09 09 09 09 09 64 65 6c 61 79  ay {.......delay
1c70: 20 3d 20 6e 65 78 74 5f 66 65 74 63 68 20 2d 20   = next_fetch - 
1c80: 6e 6f 77 3b 0a 09 09 09 09 09 7d 0a 09 09 09 09  now;......}.....
1c90: 7d 0a 09 09 09 7d 3b 0a 09 09 09 74 6f 6b 69 6f  }....};....tokio
1ca0: 3a 3a 74 69 6d 65 3a 3a 64 65 6c 61 79 5f 66 6f  ::time::delay_fo
1cb0: 72 28 64 65 6c 61 79 2e 74 6f 5f 73 74 64 28 29  r(delay.to_std()
1cc0: 3f 29 2e 61 77 61 69 74 3b 0a 09 09 7d 0a 09 09  ?).await;...}...
1cd0: 2f 2f 4f 6b 28 28 29 29 0a 09 7d 0a 0a 7d 0a 0a  //Ok(())..}..}..
1ce0: 23 5b 74 6f 6b 69 6f 3a 3a 6d 61 69 6e 28 62 61  #[tokio::main(ba
1cf0: 73 69 63 5f 73 63 68 65 64 75 6c 65 72 29 5d 0a  sic_scheduler)].
1d00: 61 73 79 6e 63 20 66 6e 20 6d 61 69 6e 28 29 20  async fn main() 
1d10: 2d 3e 20 52 65 73 75 6c 74 3c 28 29 3e 20 7b 0a  -> Result<()> {.
1d20: 09 6c 65 74 20 6d 75 74 20 73 65 74 74 69 6e 67  .let mut setting
1d30: 73 20 3d 20 63 6f 6e 66 69 67 3a 3a 43 6f 6e 66  s = config::Conf
1d40: 69 67 3a 3a 64 65 66 61 75 6c 74 28 29 3b 0a 09  ig::default();..
1d50: 73 65 74 74 69 6e 67 73 2e 6d 65 72 67 65 28 63  settings.merge(c
1d60: 6f 6e 66 69 67 3a 3a 46 69 6c 65 3a 3a 77 69 74  onfig::File::wit
1d70: 68 5f 6e 61 6d 65 28 22 72 73 73 74 67 22 29 29  h_name("rsstg"))
1d80: 3f 3b 0a 0a 09 6c 65 74 20 72 65 5f 75 73 65 72  ?;...let re_user
1d90: 6e 61 6d 65 20 3d 20 52 65 67 65 78 3a 3a 6e 65  name = Regex::ne
1da0: 77 28 72 22 5e 40 5b 61 2d 7a 41 2d 5a 5d 5b 61  w(r"^@[a-zA-Z][a
1db0: 2d 7a 41 2d 5a 30 2d 39 5f 5d 2b 24 22 29 3f 3b  -zA-Z0-9_]+$")?;
1dc0: 0a 09 6c 65 74 20 72 65 5f 6c 69 6e 6b 20 3d 20  ..let re_link = 
1dd0: 52 65 67 65 78 3a 3a 6e 65 77 28 72 22 5e 68 74  Regex::new(r"^ht
1de0: 74 70 73 3f 3a 2f 2f 5b 61 2d 7a 41 2d 5a 2e 30  tps?://[a-zA-Z.0
1df0: 2d 39 2d 5d 2b 2f 5b 2d 5f 61 2d 7a 41 2d 5a 2e  -9-]+/[-_a-zA-Z.
1e00: 30 2d 39 2f 3f 3d 5d 2b 24 22 29 3f 3b 0a 09 6c  0-9/?=]+$")?;..l
1e10: 65 74 20 72 65 5f 69 76 5f 68 61 73 68 20 3d 20  et re_iv_hash = 
1e20: 52 65 67 65 78 3a 3a 6e 65 77 28 72 22 5e 5b 61  Regex::new(r"^[a
1e30: 2d 66 30 2d 39 5d 7b 31 34 7d 24 22 29 3f 3b 0a  -f0-9]{14}$")?;.
1e40: 0a 09 6c 65 74 20 63 6f 72 65 20 3d 20 43 6f 72  ..let core = Cor
1e50: 65 3a 3a 6e 65 77 28 73 65 74 74 69 6e 67 73 29  e::new(settings)
1e60: 2e 61 77 61 69 74 3f 3b 0a 0a 09 6c 65 74 20 6d  .await?;...let m
1e70: 75 74 20 73 74 72 65 61 6d 20 3d 20 63 6f 72 65  ut stream = core
1e80: 2e 73 74 72 65 61 6d 28 29 3b 0a 0a 09 77 68 69  .stream();...whi
1e90: 6c 65 20 6c 65 74 20 53 6f 6d 65 28 75 70 64 61  le let Some(upda
1ea0: 74 65 29 20 3d 20 73 74 72 65 61 6d 2e 6e 65 78  te) = stream.nex
1eb0: 74 28 29 2e 61 77 61 69 74 20 7b 0a 09 09 6c 65  t().await {...le
1ec0: 74 20 75 70 64 61 74 65 20 3d 20 75 70 64 61 74  t update = updat
1ed0: 65 3f 3b 0a 09 09 6d 61 74 63 68 20 75 70 64 61  e?;...match upda
1ee0: 74 65 2e 6b 69 6e 64 20 7b 0a 09 09 09 55 70 64  te.kind {....Upd
1ef0: 61 74 65 4b 69 6e 64 3a 3a 4d 65 73 73 61 67 65  ateKind::Message
1f00: 28 6d 65 73 73 61 67 65 29 20 3d 3e 20 7b 0a 09  (message) => {..
1f10: 09 09 09 6c 65 74 20 6d 75 74 20 72 65 70 6c 79  ...let mut reply
1f20: 3a 20 56 65 63 3c 53 74 72 69 6e 67 3e 20 3d 20  : Vec<String> = 
1f30: 76 65 63 21 5b 5d 3b 0a 09 09 09 09 6d 61 74 63  vec![];.....matc
1f40: 68 20 6d 65 73 73 61 67 65 2e 6b 69 6e 64 20 7b  h message.kind {
1f50: 0a 09 09 09 09 09 4d 65 73 73 61 67 65 4b 69 6e  ......MessageKin
1f60: 64 3a 3a 54 65 78 74 20 7b 20 72 65 66 20 64 61  d::Text { ref da
1f70: 74 61 2c 20 2e 2e 20 7d 20 3d 3e 20 7b 0a 09 09  ta, .. } => {...
1f80: 09 09 09 09 6c 65 74 20 6d 75 74 20 77 6f 72 64  ....let mut word
1f90: 73 20 3d 20 64 61 74 61 2e 73 70 6c 69 74 5f 77  s = data.split_w
1fa0: 68 69 74 65 73 70 61 63 65 28 29 3b 0a 09 09 09  hitespace();....
1fb0: 09 09 09 6c 65 74 20 63 6d 64 20 3d 20 77 6f 72  ...let cmd = wor
1fc0: 64 73 2e 6e 65 78 74 28 29 2e 75 6e 77 72 61 70  ds.next().unwrap
1fd0: 28 29 3b 0a 09 09 09 09 09 09 6d 61 74 63 68 20  ();.......match 
1fe0: 63 6d 64 20 7b 0a 0a 2f 2f 20 73 74 61 72 74 0a  cmd {..// start.
1ff0: 0a 09 09 09 09 09 09 09 22 2f 73 74 61 72 74 22  ........"/start"
2000: 20 3d 3e 20 7b 0a 09 09 09 09 09 09 09 09 72 65   => {.........re
2010: 70 6c 79 2e 70 75 73 68 28 22 4e 6f 74 20 69 6e  ply.push("Not in
2020: 20 73 65 72 76 69 63 65 20 79 65 74 2e 20 54 72   service yet. Tr
2030: 79 20 6c 61 74 65 72 2e 22 2e 74 6f 5f 73 74 72  y later.".to_str
2040: 69 6e 67 28 29 29 3b 0a 09 09 09 09 09 09 09 7d  ing());........}
2050: 2c 0a 0a 2f 2f 20 6c 69 73 74 0a 0a 09 09 09 09  ,..// list......
2060: 09 09 09 22 2f 6c 69 73 74 22 20 3d 3e 20 7b 0a  ..."/list" => {.
2070: 09 09 09 09 09 09 09 09 72 65 70 6c 79 2e 70 75  ........reply.pu
2080: 73 68 28 22 43 68 61 6e 6e 65 6c 73 3a 22 2e 74  sh("Channels:".t
2090: 6f 5f 73 74 72 69 6e 67 28 29 29 3b 0a 09 09 09  o_string());....
20a0: 09 09 09 09 09 6c 65 74 20 6d 75 74 20 72 6f 77  .....let mut row
20b0: 73 20 3d 20 73 71 6c 78 3a 3a 71 75 65 72 79 28  s = sqlx::query(
20c0: 22 73 65 6c 65 63 74 20 75 73 65 72 6e 61 6d 65  "select username
20d0: 2c 20 65 6e 61 62 6c 65 64 2c 20 75 72 6c 2c 20  , enabled, url, 
20e0: 69 76 5f 68 61 73 68 20 66 72 6f 6d 20 72 73 73  iv_hash from rss
20f0: 74 67 5f 73 6f 75 72 63 65 20 6c 65 66 74 20 6a  tg_source left j
2100: 6f 69 6e 20 72 73 73 74 67 5f 63 68 61 6e 6e 65  oin rsstg_channe
2110: 6c 20 75 73 69 6e 67 20 28 63 68 61 6e 6e 65 6c  l using (channel
2120: 5f 69 64 29 20 77 68 65 72 65 20 6f 77 6e 65 72  _id) where owner
2130: 20 3d 20 24 31 22 29 0a 09 09 09 09 09 09 09 09   = $1").........
2140: 09 2e 62 69 6e 64 28 69 36 34 3a 3a 66 72 6f 6d  ..bind(i64::from
2150: 28 6d 65 73 73 61 67 65 2e 66 72 6f 6d 2e 69 64  (message.from.id
2160: 29 29 0a 09 09 09 09 09 09 09 09 09 2e 66 65 74  ))...........fet
2170: 63 68 28 26 63 6f 72 65 2e 70 6f 6f 6c 29 3b 0a  ch(&core.pool);.
2180: 09 09 09 09 09 09 09 09 77 68 69 6c 65 20 6c 65  ........while le
2190: 74 20 53 6f 6d 65 28 72 6f 77 29 20 3d 20 72 6f  t Some(row) = ro
21a0: 77 73 2e 74 72 79 5f 6e 65 78 74 28 29 2e 61 77  ws.try_next().aw
21b0: 61 69 74 3f 20 7b 0a 09 09 09 09 09 09 09 09 09  ait? {..........
21c0: 6c 65 74 20 75 73 65 72 6e 61 6d 65 3a 20 26 73  let username: &s
21d0: 74 72 20 3d 20 72 6f 77 2e 74 72 79 5f 67 65 74  tr = row.try_get
21e0: 28 22 75 73 65 72 6e 61 6d 65 22 29 3f 3b 0a 09  ("username")?;..
21f0: 09 09 09 09 09 09 09 09 6c 65 74 20 65 6e 61 62  ........let enab
2200: 6c 65 64 3a 20 62 6f 6f 6c 20 3d 20 72 6f 77 2e  led: bool = row.
2210: 74 72 79 5f 67 65 74 28 22 65 6e 61 62 6c 65 64  try_get("enabled
2220: 22 29 3f 3b 0a 09 09 09 09 09 09 09 09 09 6c 65  ")?;..........le
2230: 74 20 75 72 6c 3a 20 26 73 74 72 20 3d 20 72 6f  t url: &str = ro
2240: 77 2e 74 72 79 5f 67 65 74 28 22 75 72 6c 22 29  w.try_get("url")
2250: 3f 3b 0a 09 09 09 09 09 09 09 09 09 6c 65 74 20  ?;..........let 
2260: 69 76 5f 68 61 73 68 3a 20 4f 70 74 69 6f 6e 3c  iv_hash: Option<
2270: 26 73 74 72 3e 20 3d 20 72 6f 77 2e 74 72 79 5f  &str> = row.try_
2280: 67 65 74 28 22 69 76 5f 68 61 73 68 22 29 3f 3b  get("iv_hash")?;
2290: 0a 09 09 09 09 09 09 09 09 09 72 65 70 6c 79 2e  ..........reply.
22a0: 70 75 73 68 28 66 6f 72 6d 61 74 21 28 22 5c 6e  push(format!("\n
22b0: 5c 5c 2a ef b8 8f e2 83 a3 20 60 7b 7d 60 20 7b  \\*ļøāƒ£ `{}` {
22c0: 7d 5c 6e f0 9f 94 97 20 60 7b 7d 60 22 2c 20 75  }\nšŸ”— `{}`", u
22d0: 73 65 72 6e 61 6d 65 2c 20 20 0a 09 09 09 09 09  sername,  ......
22e0: 09 09 09 09 09 6d 61 74 63 68 20 65 6e 61 62 6c  .....match enabl
22f0: 65 64 20 7b 0a 09 09 09 09 09 09 09 09 09 09 09  ed {............
2300: 74 72 75 65 20 20 3d 3e 20 22 f0 9f 94 84 20 65  true  => "šŸ”„ e
2310: 6e 61 62 6c 65 64 22 2c 0a 09 09 09 09 09 09 09  nabled",........
2320: 09 09 09 09 66 61 6c 73 65 20 3d 3e 20 22 e2 9b  ....false => "ā›
2330: 94 20 64 69 73 61 62 6c 65 64 22 2c 0a 09 09 09  ” disabled",....
2340: 09 09 09 09 09 09 09 7d 2c 20 75 72 6c 29 29 3b  .......}, url));
2350: 0a 09 09 09 09 09 09 09 09 09 69 66 20 6c 65 74  ..........if let
2360: 20 53 6f 6d 65 28 68 61 73 68 29 20 3d 20 69 76   Some(hash) = iv
2370: 5f 68 61 73 68 20 7b 0a 09 09 09 09 09 09 09 09  _hash {.........
2380: 09 09 72 65 70 6c 79 2e 70 75 73 68 28 66 6f 72  ..reply.push(for
2390: 6d 61 74 21 28 22 49 56 20 60 7b 7d 60 22 2c 20  mat!("IV `{}`", 
23a0: 68 61 73 68 29 29 3b 0a 09 09 09 09 09 09 09 09  hash));.........
23b0: 09 7d 0a 09 09 09 09 09 09 09 09 7d 0a 09 09 09  .}.........}....
23c0: 09 09 09 09 7d 2c 0a 0a 2f 2f 20 61 64 64 0a 0a  ....},..// add..
23d0: 09 09 09 09 09 09 09 22 2f 61 64 64 22 20 3d 3e  ......."/add" =>
23e0: 20 7b 0a 09 09 09 09 09 09 09 09 6c 65 74 20 28   {.........let (
23f0: 63 68 61 6e 6e 65 6c 2c 20 75 72 6c 2c 20 69 76  channel, url, iv
2400: 5f 68 61 73 68 29 20 3d 20 28 77 6f 72 64 73 2e  _hash) = (words.
2410: 6e 65 78 74 28 29 2e 75 6e 77 72 61 70 28 29 2c  next().unwrap(),
2420: 20 77 6f 72 64 73 2e 6e 65 78 74 28 29 2e 75 6e   words.next().un
2430: 77 72 61 70 28 29 2c 20 77 6f 72 64 73 2e 6e 65  wrap(), words.ne
2440: 78 74 28 29 29 3b 0a 09 09 09 09 09 09 09 09 6c  xt());.........l
2450: 65 74 20 6f 6b 5f 6c 69 6e 6b 20 3d 20 72 65 5f  et ok_link = re_
2460: 6c 69 6e 6b 2e 69 73 5f 6d 61 74 63 68 28 26 75  link.is_match(&u
2470: 72 6c 29 3b 0a 09 09 09 09 09 09 09 09 6c 65 74  rl);.........let
2480: 20 6f 6b 5f 68 61 73 68 20 3d 20 6d 61 74 63 68   ok_hash = match
2490: 20 69 76 5f 68 61 73 68 20 7b 0a 09 09 09 09 09   iv_hash {......
24a0: 09 09 09 09 53 6f 6d 65 28 68 61 73 68 29 20 3d  ....Some(hash) =
24b0: 3e 20 72 65 5f 69 76 5f 68 61 73 68 2e 69 73 5f  > re_iv_hash.is_
24c0: 6d 61 74 63 68 28 26 68 61 73 68 29 2c 0a 09 09  match(&hash),...
24d0: 09 09 09 09 09 09 09 4e 6f 6e 65 20 3d 3e 20 74  .......None => t
24e0: 72 75 65 2c 0a 09 09 09 09 09 09 09 09 7d 3b 0a  rue,.........};.
24f0: 09 09 09 09 09 09 09 09 69 66 20 21 20 6f 6b 5f  ........if ! ok_
2500: 6c 69 6e 6b 20 7b 0a 09 09 09 09 09 09 09 09 09  link {..........
2510: 72 65 70 6c 79 2e 70 75 73 68 28 22 4c 69 6e 6b  reply.push("Link
2520: 20 73 68 6f 75 6c 64 20 62 65 20 6c 69 6e 6b 20   should be link 
2530: 74 6f 20 61 74 6f 6d 2f 72 73 73 20 66 65 65 64  to atom/rss feed
2540: 2c 20 73 6f 6d 65 74 68 69 6e 67 20 6c 69 6b 65  , something like
2550: 20 5c 22 68 74 74 70 73 3a 2f 2f 64 6f 6d 61 69   \"https://domai
2560: 6e 2f 70 61 74 68 5c 22 5c 5c 2e 22 2e 74 6f 5f  n/path\"\\.".to_
2570: 73 74 72 69 6e 67 28 29 29 3b 0a 09 09 09 09 09  string());......
2580: 09 09 09 09 63 6f 72 65 2e 64 65 62 75 67 28 26  ....core.debug(&
2590: 66 6f 72 6d 61 74 21 28 22 55 72 6c 3a 20 7b 3a  format!("Url: {:
25a0: 3f 7d 22 2c 20 26 75 72 6c 29 29 3f 3b 0a 09 09  ?}", &url))?;...
25b0: 09 09 09 09 09 09 7d 0a 09 09 09 09 09 09 09 09  ......}.........
25c0: 69 66 20 21 20 6f 6b 5f 68 61 73 68 20 7b 0a 09  if ! ok_hash {..
25d0: 09 09 09 09 09 09 09 09 72 65 70 6c 79 2e 70 75  ........reply.pu
25e0: 73 68 28 22 49 56 20 68 61 73 68 20 73 68 6f 75  sh("IV hash shou
25f0: 6c 64 20 62 65 20 31 34 20 68 65 78 20 64 69 67  ld be 14 hex dig
2600: 69 74 73 2e 22 2e 74 6f 5f 73 74 72 69 6e 67 28  its.".to_string(
2610: 29 29 3b 0a 09 09 09 09 09 09 09 09 09 63 6f 72  ));..........cor
2620: 65 2e 64 65 62 75 67 28 26 66 6f 72 6d 61 74 21  e.debug(&format!
2630: 28 22 49 56 3a 20 7b 3a 3f 7d 22 2c 20 26 69 76  ("IV: {:?}", &iv
2640: 5f 68 61 73 68 29 29 3f 3b 0a 09 09 09 09 09 09  _hash))?;.......
2650: 09 09 7d 0a 09 09 09 09 09 09 09 09 69 66 20 6f  ..}.........if o
2660: 6b 5f 6c 69 6e 6b 20 26 26 20 6f 6b 5f 68 61 73  k_link && ok_has
2670: 68 20 7b 0a 09 09 09 09 09 09 09 09 09 6c 65 74  h {..........let
2680: 20 63 68 61 6e 3a 20 4f 70 74 69 6f 6e 3c 69 36   chan: Option<i6
2690: 34 3e 20 3d 20 6d 61 74 63 68 20 73 71 6c 78 3a  4> = match sqlx:
26a0: 3a 71 75 65 72 79 28 22 73 65 6c 65 63 74 20 63  :query("select c
26b0: 68 61 6e 6e 65 6c 5f 69 64 20 66 72 6f 6d 20 72  hannel_id from r
26c0: 73 73 74 67 5f 63 68 61 6e 6e 65 6c 20 77 68 65  sstg_channel whe
26d0: 72 65 20 75 73 65 72 6e 61 6d 65 20 3d 20 24 31  re username = $1
26e0: 22 29 0a 09 09 09 09 09 09 09 09 09 09 2e 62 69  ")............bi
26f0: 6e 64 28 63 68 61 6e 6e 65 6c 29 0a 09 09 09 09  nd(channel).....
2700: 09 09 09 09 09 09 2e 66 65 74 63 68 5f 6f 6e 65  .......fetch_one
2710: 28 26 63 6f 72 65 2e 70 6f 6f 6c 29 2e 61 77 61  (&core.pool).awa
2720: 69 74 20 7b 0a 09 09 09 09 09 09 09 09 09 09 09  it {............
2730: 4f 6b 28 63 68 61 6e 29 20 3d 3e 20 53 6f 6d 65  Ok(chan) => Some
2740: 28 63 68 61 6e 2e 74 72 79 5f 67 65 74 28 22 63  (chan.try_get("c
2750: 68 61 6e 6e 65 6c 5f 69 64 22 29 3f 29 2c 0a 09  hannel_id")?),..
2760: 09 09 09 09 09 09 09 09 09 09 45 72 72 28 73 71  ..........Err(sq
2770: 6c 78 3a 3a 45 72 72 6f 72 3a 3a 52 6f 77 4e 6f  lx::Error::RowNo
2780: 74 46 6f 75 6e 64 29 20 3d 3e 20 7b 0a 09 09 09  tFound) => {....
2790: 09 09 09 09 09 09 09 09 09 72 65 70 6c 79 2e 70  .........reply.p
27a0: 75 73 68 28 22 53 6f 72 72 79 2c 20 49 20 64 6f  ush("Sorry, I do
27b0: 6e 27 74 20 6b 6e 6f 77 20 61 62 6f 75 74 20 74  n't know about t
27c0: 68 61 74 20 63 68 61 6e 6e 65 6c 2e 20 50 6c 65  hat channel. Ple
27d0: 61 73 65 2c 20 61 64 64 20 61 20 63 68 61 6e 6e  ase, add a chann
27e0: 65 6c 20 77 69 74 68 20 2f 61 64 64 63 68 61 6e  el with /addchan
27f0: 2e 22 2e 74 6f 5f 73 74 72 69 6e 67 28 29 29 3b  .".to_string());
2800: 0a 09 09 09 09 09 09 09 09 09 09 09 09 4e 6f 6e  .............Non
2810: 65 0a 09 09 09 09 09 09 09 09 09 09 09 7d 2c 0a  e............},.
2820: 09 09 09 09 09 09 09 09 09 09 09 45 72 72 28 65  ...........Err(e
2830: 72 72 29 20 3d 3e 20 7b 0a 09 09 09 09 09 09 09  rr) => {........
2840: 09 09 09 09 09 72 65 70 6c 79 2e 70 75 73 68 28  .....reply.push(
2850: 22 53 6f 72 72 79 2c 20 75 6e 6b 6e 6f 77 6e 20  "Sorry, unknown 
2860: 65 72 72 6f 72 5c 5c 2e 22 2e 74 6f 5f 73 74 72  error\\.".to_str
2870: 69 6e 67 28 29 29 3b 0a 09 09 09 09 09 09 09 09  ing());.........
2880: 09 09 09 09 63 6f 72 65 2e 64 65 62 75 67 28 26  ....core.debug(&
2890: 66 6f 72 6d 61 74 21 28 22 53 6f 72 72 79 2c 20  format!("Sorry, 
28a0: 75 6e 6b 6e 6f 77 6e 20 65 72 72 6f 72 3a 20 7b  unknown error: {
28b0: 3a 23 3f 7d 5c 6e 22 2c 20 65 72 72 29 29 3f 3b  :#?}\n", err))?;
28c0: 0a 09 09 09 09 09 09 09 09 09 09 09 09 4e 6f 6e  .............Non
28d0: 65 0a 09 09 09 09 09 09 09 09 09 09 09 7d 2c 0a  e............},.
28e0: 09 09 09 09 09 09 09 09 09 7d 3b 0a 09 09 09 09  .........};.....
28f0: 09 09 09 09 09 6d 61 74 63 68 20 63 68 61 6e 20  .....match chan 
2900: 7b 0a 09 09 09 09 09 09 09 09 09 09 53 6f 6d 65  {...........Some
2910: 28 63 68 61 6e 29 20 3d 3e 20 7b 0a 09 09 09 09  (chan) => {.....
2920: 09 09 09 09 09 09 09 6d 61 74 63 68 20 73 71 6c  .......match sql
2930: 78 3a 3a 71 75 65 72 79 28 22 69 6e 73 65 72 74  x::query("insert
2940: 20 69 6e 74 6f 20 72 73 73 74 67 5f 73 6f 75 72   into rsstg_sour
2950: 63 65 20 28 63 68 61 6e 6e 65 6c 5f 69 64 2c 20  ce (channel_id, 
2960: 75 72 6c 2c 20 69 76 5f 68 61 73 68 2c 20 6f 77  url, iv_hash, ow
2970: 6e 65 72 29 20 76 61 6c 75 65 73 20 28 24 31 2c  ner) values ($1,
2980: 20 24 32 2c 20 24 33 2c 20 24 34 29 20 6f 6e 20   $2, $3, $4) on 
2990: 63 6f 6e 66 6c 69 63 74 20 28 63 68 61 6e 6e 65  conflict (channe
29a0: 6c 5f 69 64 2c 20 6f 77 6e 65 72 29 20 64 6f 20  l_id, owner) do 
29b0: 75 70 64 61 74 65 20 73 65 74 20 75 72 6c 20 3d  update set url =
29c0: 20 65 78 63 6c 75 64 65 64 2e 75 72 6c 2c 20 69   excluded.url, i
29d0: 76 5f 68 61 73 68 20 3d 20 65 78 63 6c 75 64 65  v_hash = exclude
29e0: 64 2e 69 76 5f 68 61 73 68 3b 22 29 0a 09 09 09  d.iv_hash;")....
29f0: 09 09 09 09 09 09 09 09 09 2e 62 69 6e 64 28 63  ..........bind(c
2a00: 68 61 6e 29 0a 09 09 09 09 09 09 09 09 09 09 09  han)............
2a10: 09 2e 62 69 6e 64 28 75 72 6c 29 0a 09 09 09 09  ..bind(url).....
2a20: 09 09 09 09 09 09 09 09 2e 62 69 6e 64 28 69 76  .........bind(iv
2a30: 5f 68 61 73 68 29 0a 09 09 09 09 09 09 09 09 09  _hash)..........
2a40: 09 09 09 2e 62 69 6e 64 28 69 36 34 3a 3a 66 72  ....bind(i64::fr
2a50: 6f 6d 28 6d 65 73 73 61 67 65 2e 66 72 6f 6d 2e  om(message.from.
2a60: 69 64 29 29 0a 09 09 09 09 09 09 09 09 09 09 09  id))............
2a70: 09 2e 65 78 65 63 75 74 65 28 26 63 6f 72 65 2e  ..execute(&core.
2a80: 70 6f 6f 6c 29 2e 61 77 61 69 74 20 7b 0a 09 09  pool).await {...
2a90: 09 09 09 09 09 09 09 09 09 09 4f 6b 28 5f 29 20  ..........Ok(_) 
2aa0: 3d 3e 20 72 65 70 6c 79 2e 70 75 73 68 28 22 43  => reply.push("C
2ab0: 68 61 6e 6e 65 6c 20 61 64 64 65 64 5c 5c 2e 22  hannel added\\."
2ac0: 2e 74 6f 5f 73 74 72 69 6e 67 28 29 29 2c 0a 09  .to_string()),..
2ad0: 09 09 09 09 09 09 09 09 09 09 09 45 72 72 28 73  ...........Err(s
2ae0: 71 6c 78 3a 3a 45 72 72 6f 72 3a 3a 44 61 74 61  qlx::Error::Data
2af0: 62 61 73 65 28 65 72 72 29 29 20 3d 3e 20 7b 0a  base(err)) => {.
2b00: 09 09 09 09 09 09 09 09 09 09 09 09 09 6d 61 74  .............mat
2b10: 63 68 20 65 72 72 2e 64 6f 77 6e 63 61 73 74 3a  ch err.downcast:
2b20: 3a 3c 73 71 6c 78 3a 3a 70 6f 73 74 67 72 65 73  :<sqlx::postgres
2b30: 3a 3a 50 67 44 61 74 61 62 61 73 65 45 72 72 6f  ::PgDatabaseErro
2b40: 72 3e 28 29 2e 72 6f 75 74 69 6e 65 28 29 20 7b  r>().routine() {
2b50: 0a 09 09 09 09 09 09 09 09 09 09 09 09 09 09 53  ...............S
2b60: 6f 6d 65 28 22 5f 62 74 5f 63 68 65 63 6b 5f 75  ome("_bt_check_u
2b70: 6e 69 71 75 65 22 2c 20 29 20 3d 3e 20 7b 0a 09  nique", ) => {..
2b80: 09 09 09 09 09 09 09 09 09 09 09 09 09 09 72 65  ..............re
2b90: 70 6c 79 2e 70 75 73 68 28 22 44 75 70 6c 69 63  ply.push("Duplic
2ba0: 61 74 65 20 6b 65 79 5c 5c 2e 22 2e 74 6f 5f 73  ate key\\.".to_s
2bb0: 74 72 69 6e 67 28 29 29 3b 0a 09 09 09 09 09 09  tring());.......
2bc0: 09 09 09 09 09 09 09 09 7d 2c 0a 09 09 09 09 09  ........},......
2bd0: 09 09 09 09 09 09 09 09 09 53 6f 6d 65 28 5f 29  .........Some(_)
2be0: 20 3d 3e 20 7b 0a 09 09 09 09 09 09 09 09 09 09   => {...........
2bf0: 09 09 09 09 09 72 65 70 6c 79 2e 70 75 73 68 28  .....reply.push(
2c00: 22 44 61 74 61 62 61 73 65 20 65 72 72 6f 72 5c  "Database error\
2c10: 5c 2e 22 2e 74 6f 5f 73 74 72 69 6e 67 28 29 29  \.".to_string())
2c20: 3b 0a 09 09 09 09 09 09 09 09 09 09 09 09 09 09  ;...............
2c30: 7d 2c 0a 09 09 09 09 09 09 09 09 09 09 09 09 09  },..............
2c40: 09 4e 6f 6e 65 20 3d 3e 20 7b 0a 09 09 09 09 09  .None => {......
2c50: 09 09 09 09 09 09 09 09 09 09 72 65 70 6c 79 2e  ..........reply.
2c60: 70 75 73 68 28 22 4e 6f 20 64 61 74 61 62 61 73  push("No databas
2c70: 65 20 65 72 72 6f 72 20 65 78 74 72 61 63 74 65  e error extracte
2c80: 64 5c 5c 2e 22 2e 74 6f 5f 73 74 72 69 6e 67 28  d\\.".to_string(
2c90: 29 29 3b 0a 09 09 09 09 09 09 09 09 09 09 09 09  ));.............
2ca0: 09 09 7d 2c 0a 09 09 09 09 09 09 09 09 09 09 09  ..},............
2cb0: 09 09 7d 3b 0a 09 09 09 09 09 09 09 09 09 09 09  ..};............
2cc0: 09 7d 2c 0a 09 09 09 09 09 09 09 09 09 09 09 09  .},.............
2cd0: 45 72 72 28 65 72 72 29 20 3d 3e 20 7b 0a 09 09  Err(err) => {...
2ce0: 09 09 09 09 09 09 09 09 09 09 09 72 65 70 6c 79  ...........reply
2cf0: 2e 70 75 73 68 28 22 53 6f 72 72 79 2c 20 75 6e  .push("Sorry, un
2d00: 6b 6e 6f 77 6e 20 65 72 72 6f 72 5c 5c 2e 22 2e  known error\\.".
2d10: 74 6f 5f 73 74 72 69 6e 67 28 29 29 3b 0a 09 09  to_string());...
2d20: 09 09 09 09 09 09 09 09 09 09 09 63 6f 72 65 2e  ...........core.
2d30: 64 65 62 75 67 28 26 66 6f 72 6d 61 74 21 28 22  debug(&format!("
2d40: 53 6f 72 72 79 2c 20 75 6e 6b 6e 6f 77 6e 20 65  Sorry, unknown e
2d50: 72 72 6f 72 3a 20 7b 3a 23 3f 7d 5c 6e 22 2c 20  rror: {:#?}\n", 
2d60: 65 72 72 29 29 3f 3b 0a 09 09 09 09 09 09 09 09  err))?;.........
2d70: 09 09 09 09 7d 2c 0a 09 09 09 09 09 09 09 09 09  ....},..........
2d80: 09 09 7d 3b 0a 09 09 09 09 09 09 09 09 09 09 7d  ..};...........}
2d90: 2c 0a 09 09 09 09 09 09 09 09 09 09 4e 6f 6e 65  ,...........None
2da0: 20 3d 3e 20 7b 7d 2c 0a 09 09 09 09 09 09 09 09   => {},.........
2db0: 09 7d 3b 0a 09 09 09 09 09 09 09 09 7d 3b 0a 09  .};.........};..
2dc0: 09 09 09 09 09 09 7d 2c 0a 0a 2f 2f 20 61 64 64  ......},..// add
2dd0: 63 68 61 6e 0a 0a 09 09 09 09 09 09 09 22 2f 61  chan........."/a
2de0: 64 64 63 68 61 6e 22 20 3d 3e 20 7b 0a 09 09 09  ddchan" => {....
2df0: 09 09 09 09 09 6c 65 74 20 63 68 61 6e 6e 65 6c  .....let channel
2e00: 20 3d 20 77 6f 72 64 73 2e 6e 65 78 74 28 29 2e   = words.next().
2e10: 75 6e 77 72 61 70 28 29 3b 0a 09 09 09 09 09 09  unwrap();.......
2e20: 09 09 69 66 20 21 20 72 65 5f 75 73 65 72 6e 61  ..if ! re_userna
2e30: 6d 65 2e 69 73 5f 6d 61 74 63 68 28 26 63 68 61  me.is_match(&cha
2e40: 6e 6e 65 6c 29 20 7b 0a 09 09 09 09 09 09 09 09  nnel) {.........
2e50: 09 72 65 70 6c 79 2e 70 75 73 68 28 22 55 73 65  .reply.push("Use
2e60: 72 6e 61 6d 65 73 20 73 68 6f 75 6c 64 20 62 65  rnames should be
2e70: 20 73 6f 6d 65 74 68 69 6e 67 20 6c 69 6b 65 20   something like 
2e80: 5c 22 40 5c 5c 5b 61 5c 5c 2d 7a 41 5c 5c 2d 5a  \"@\\[a\\-zA\\-Z
2e90: 5d 5c 5c 5b 61 5c 5c 2d 7a 41 5c 5c 2d 5a 30 5c  ]\\[a\\-zA\\-Z0\
2ea0: 5c 2d 39 5c 5c 5f 5d 2b 5c 22 2c 20 61 72 65 6e  \-9\\_]+\", aren
2eb0: 27 74 20 74 68 65 79 3f 22 2e 74 6f 5f 73 74 72  't they?".to_str
2ec0: 69 6e 67 28 29 29 3b 0a 09 09 09 09 09 09 09 09  ing());.........
2ed0: 7d 20 65 6c 73 65 20 7b 0a 09 09 09 09 09 09 09  } else {........
2ee0: 09 09 6c 65 74 20 63 68 61 6e 3a 20 4f 70 74 69  ..let chan: Opti
2ef0: 6f 6e 3c 69 36 34 3e 20 3d 20 6d 61 74 63 68 20  on<i64> = match 
2f00: 73 71 6c 78 3a 3a 71 75 65 72 79 28 22 73 65 6c  sqlx::query("sel
2f10: 65 63 74 20 63 68 61 6e 6e 65 6c 5f 69 64 20 66  ect channel_id f
2f20: 72 6f 6d 20 72 73 73 74 67 5f 63 68 61 6e 6e 65  rom rsstg_channe
2f30: 6c 20 77 68 65 72 65 20 75 73 65 72 6e 61 6d 65  l where username
2f40: 20 3d 20 24 31 22 29 0a 09 09 09 09 09 09 09 09   = $1").........
2f50: 09 09 2e 62 69 6e 64 28 63 68 61 6e 6e 65 6c 29  ...bind(channel)
2f60: 0a 09 09 09 09 09 09 09 09 09 09 2e 66 65 74 63  ............fetc
2f70: 68 5f 6f 6e 65 28 26 63 6f 72 65 2e 70 6f 6f 6c  h_one(&core.pool
2f80: 29 2e 61 77 61 69 74 20 7b 0a 09 09 09 09 09 09  ).await {.......
2f90: 09 09 09 09 09 4f 6b 28 63 68 61 6e 29 20 3d 3e  .....Ok(chan) =>
2fa0: 20 53 6f 6d 65 28 63 68 61 6e 2e 74 72 79 5f 67   Some(chan.try_g
2fb0: 65 74 28 22 63 68 61 6e 6e 65 6c 5f 69 64 22 29  et("channel_id")
2fc0: 3f 29 2c 0a 09 09 09 09 09 09 09 09 09 09 09 45  ?),............E
2fd0: 72 72 28 73 71 6c 78 3a 3a 45 72 72 6f 72 3a 3a  rr(sqlx::Error::
2fe0: 52 6f 77 4e 6f 74 46 6f 75 6e 64 29 20 3d 3e 20  RowNotFound) => 
2ff0: 4e 6f 6e 65 2c 0a 09 09 09 09 09 09 09 09 09 09  None,...........
3000: 09 45 72 72 28 65 72 72 29 20 3d 3e 20 7b 0a 09  .Err(err) => {..
3010: 09 09 09 09 09 09 09 09 09 09 09 72 65 70 6c 79  ...........reply
3020: 2e 70 75 73 68 28 22 53 6f 72 72 79 2c 20 75 6e  .push("Sorry, un
3030: 6b 6e 6f 77 6e 20 65 72 72 6f 72 5c 5c 2e 22 2e  known error\\.".
3040: 74 6f 5f 73 74 72 69 6e 67 28 29 29 3b 0a 09 09  to_string());...
3050: 09 09 09 09 09 09 09 09 09 09 63 6f 72 65 2e 64  ..........core.d
3060: 65 62 75 67 28 26 66 6f 72 6d 61 74 21 28 22 53  ebug(&format!("S
3070: 6f 72 72 79 2c 20 75 6e 6b 6e 6f 77 6e 20 65 72  orry, unknown er
3080: 72 6f 72 3a 20 7b 3a 23 3f 7d 5c 6e 22 2c 20 65  ror: {:#?}\n", e
3090: 72 72 29 29 3f 3b 0a 09 09 09 09 09 09 09 09 09  rr))?;..........
30a0: 09 09 09 4e 6f 6e 65 0a 09 09 09 09 09 09 09 09  ...None.........
30b0: 09 09 09 7d 2c 0a 09 09 09 09 09 09 09 09 09 7d  ...},..........}
30c0: 3b 0a 09 09 09 09 09 09 09 09 09 6d 61 74 63 68  ;..........match
30d0: 20 63 68 61 6e 20 7b 0a 09 09 09 09 09 09 09 09   chan {.........
30e0: 09 09 53 6f 6d 65 28 63 68 61 6e 29 20 3d 3e 20  ..Some(chan) => 
30f0: 7b 0a 09 09 09 09 09 09 09 09 09 09 09 6c 65 74  {............let
3100: 20 6e 65 77 5f 63 68 61 6e 20 3d 20 63 6f 72 65   new_chan = core
3110: 2e 74 67 2e 73 65 6e 64 28 74 65 6c 65 67 72 61  .tg.send(telegra
3120: 6d 5f 62 6f 74 3a 3a 47 65 74 43 68 61 74 3a 3a  m_bot::GetChat::
3130: 6e 65 77 28 74 65 6c 65 67 72 61 6d 5f 62 6f 74  new(telegram_bot
3140: 3a 3a 74 79 70 65 73 3a 3a 43 68 61 74 49 64 3a  ::types::ChatId:
3150: 3a 6e 65 77 28 63 68 61 6e 29 29 29 2e 61 77 61  :new(chan))).awa
3160: 69 74 3f 3b 0a 09 09 09 09 09 09 09 09 09 09 09  it?;............
3170: 69 66 20 69 36 34 3a 3a 66 72 6f 6d 28 6e 65 77  if i64::from(new
3180: 5f 63 68 61 6e 2e 69 64 28 29 29 20 3d 3d 20 63  _chan.id()) == c
3190: 68 61 6e 20 7b 0a 09 09 09 09 09 09 09 09 09 09  han {...........
31a0: 09 09 72 65 70 6c 79 2e 70 75 73 68 28 22 49 20  ..reply.push("I 
31b0: 61 6c 72 65 61 64 79 20 6b 6e 6f 77 20 74 68 61  already know tha
31c0: 74 20 63 68 61 6e 6e 65 6c 5c 5c 2e 22 2e 74 6f  t channel\\.".to
31d0: 5f 73 74 72 69 6e 67 28 29 29 3b 0a 09 09 09 09  _string());.....
31e0: 09 09 09 09 09 09 09 7d 20 65 6c 73 65 20 7b 0a  .......} else {.
31f0: 09 09 09 09 09 09 09 09 09 09 09 09 72 65 70 6c  ............repl
3200: 79 2e 70 75 73 68 28 22 48 6d 6d 2c 20 63 68 61  y.push("Hmm, cha
3210: 6e 6e 65 6c 20 68 61 73 20 63 68 61 6e 67 65 64  nnel has changed
3220: e2 80 a6 20 49 27 6c 6c 20 66 69 78 20 69 74 20  … I'll fix it 
3230: 6c 61 74 65 72 5c 5c 2e 22 2e 74 6f 5f 73 74 72  later\\.".to_str
3240: 69 6e 67 28 29 29 3b 0a 09 09 09 09 09 09 09 09  ing());.........
3250: 09 09 09 7d 3b 0a 09 09 09 09 09 09 09 09 09 09  ...};...........
3260: 7d 2c 0a 09 09 09 09 09 09 09 09 09 09 4e 6f 6e  },...........Non
3270: 65 20 3d 3e 20 7b 0a 09 09 09 09 09 09 09 09 09  e => {..........
3280: 09 09 6d 61 74 63 68 20 63 6f 72 65 2e 74 67 2e  ..match core.tg.
3290: 73 65 6e 64 28 74 65 6c 65 67 72 61 6d 5f 62 6f  send(telegram_bo
32a0: 74 3a 3a 47 65 74 43 68 61 74 41 64 6d 69 6e 69  t::GetChatAdmini
32b0: 73 74 72 61 74 6f 72 73 3a 3a 6e 65 77 28 74 65  strators::new(te
32c0: 6c 65 67 72 61 6d 5f 62 6f 74 3a 3a 74 79 70 65  legram_bot::type
32d0: 73 3a 3a 43 68 61 74 52 65 66 3a 3a 43 68 61 6e  s::ChatRef::Chan
32e0: 6e 65 6c 55 73 65 72 6e 61 6d 65 28 63 68 61 6e  nelUsername(chan
32f0: 6e 65 6c 2e 74 6f 5f 73 74 72 69 6e 67 28 29 29  nel.to_string())
3300: 29 29 2e 61 77 61 69 74 20 7b 0a 09 09 09 09 09  )).await {......
3310: 09 09 09 09 09 09 09 4f 6b 28 63 68 61 6e 5f 61  .......Ok(chan_a
3320: 64 6d 29 20 3d 3e 20 7b 0a 09 09 09 09 09 09 09  dm) => {........
3330: 09 09 09 09 09 09 6c 65 74 20 28 6d 75 74 20 6d  ......let (mut m
3340: 65 2c 20 6d 75 74 20 75 73 65 72 29 20 3d 20 28  e, mut user) = (
3350: 66 61 6c 73 65 2c 20 66 61 6c 73 65 29 3b 0a 09  false, false);..
3360: 09 09 09 09 09 09 09 09 09 09 09 09 66 6f 72 20  ............for 
3370: 61 64 6d 69 6e 20 69 6e 20 26 63 68 61 6e 5f 61  admin in &chan_a
3380: 64 6d 20 7b 0a 09 09 09 09 09 09 09 09 09 09 09  dm {............
3390: 09 09 09 69 66 20 61 64 6d 69 6e 2e 75 73 65 72  ...if admin.user
33a0: 2e 69 64 20 3d 3d 20 63 6f 72 65 2e 6d 79 2e 69  .id == core.my.i
33b0: 64 20 7b 0a 09 09 09 09 09 09 09 09 09 09 09 09  d {.............
33c0: 09 09 09 6d 65 20 3d 20 74 72 75 65 3b 0a 09 09  ...me = true;...
33d0: 09 09 09 09 09 09 09 09 09 09 09 09 7d 3b 0a 09  ............};..
33e0: 09 09 09 09 09 09 09 09 09 09 09 09 09 69 66 20  .............if 
33f0: 61 64 6d 69 6e 2e 75 73 65 72 2e 69 64 20 3d 3d  admin.user.id ==
3400: 20 6d 65 73 73 61 67 65 2e 66 72 6f 6d 2e 69 64   message.from.id
3410: 20 7b 0a 09 09 09 09 09 09 09 09 09 09 09 09 09   {..............
3420: 09 09 75 73 65 72 20 3d 20 74 72 75 65 3b 0a 09  ..user = true;..
3430: 09 09 09 09 09 09 09 09 09 09 09 09 09 7d 3b 0a  .............};.
3440: 09 09 09 09 09 09 09 09 09 09 09 09 09 7d 3b 0a  .............};.
3450: 09 09 09 09 09 09 09 09 09 09 09 09 09 69 66 20  .............if 
3460: 21 20 6d 65 20 20 20 7b 20 72 65 70 6c 79 2e 70  ! me   { reply.p
3470: 75 73 68 28 22 49 20 6e 65 65 64 20 74 6f 20 62  ush("I need to b
3480: 65 20 61 64 6d 69 6e 20 6f 6e 20 74 68 61 74 20  e admin on that 
3490: 63 68 61 6e 6e 65 6c 5c 5c 2e 22 2e 74 6f 5f 73  channel\\.".to_s
34a0: 74 72 69 6e 67 28 29 29 3b 20 7d 3b 0a 09 09 09  tring()); };....
34b0: 09 09 09 09 09 09 09 09 09 09 69 66 20 21 20 75  ..........if ! u
34c0: 73 65 72 20 7b 20 72 65 70 6c 79 2e 70 75 73 68  ser { reply.push
34d0: 28 22 59 6f 75 20 73 68 6f 75 6c 64 20 62 65 20  ("You should be 
34e0: 61 64 6d 69 6e 20 6f 6e 20 74 68 61 74 20 63 68  admin on that ch
34f0: 61 6e 6e 65 6c 5c 5c 2e 22 2e 74 6f 5f 73 74 72  annel\\.".to_str
3500: 69 6e 67 28 29 29 3b 20 7d 3b 0a 09 09 09 09 09  ing()); };......
3510: 09 09 09 09 09 09 09 09 69 66 20 6d 65 20 26 26  ........if me &&
3520: 20 75 73 65 72 20 7b 0a 09 09 09 09 09 09 09 09   user {.........
3530: 09 09 09 09 09 09 6c 65 74 20 63 68 61 6e 5f 69  ......let chan_i
3540: 64 20 3d 20 63 6f 72 65 2e 74 67 2e 73 65 6e 64  d = core.tg.send
3550: 28 74 65 6c 65 67 72 61 6d 5f 62 6f 74 3a 3a 47  (telegram_bot::G
3560: 65 74 43 68 61 74 3a 3a 6e 65 77 28 74 65 6c 65  etChat::new(tele
3570: 67 72 61 6d 5f 62 6f 74 3a 3a 74 79 70 65 73 3a  gram_bot::types:
3580: 3a 43 68 61 74 52 65 66 3a 3a 43 68 61 6e 6e 65  :ChatRef::Channe
3590: 6c 55 73 65 72 6e 61 6d 65 28 63 68 61 6e 6e 65  lUsername(channe
35a0: 6c 2e 74 6f 5f 73 74 72 69 6e 67 28 29 29 29 29  l.to_string())))
35b0: 2e 61 77 61 69 74 3f 3b 0a 09 09 09 09 09 09 09  .await?;........
35c0: 09 09 09 09 09 09 09 73 71 6c 78 3a 3a 71 75 65  .......sqlx::que
35d0: 72 79 28 22 69 6e 73 65 72 74 20 69 6e 74 6f 20  ry("insert into 
35e0: 72 73 73 74 67 5f 63 68 61 6e 6e 65 6c 20 28 63  rsstg_channel (c
35f0: 68 61 6e 6e 65 6c 5f 69 64 2c 20 75 73 65 72 6e  hannel_id, usern
3600: 61 6d 65 29 20 76 61 6c 75 65 73 20 28 24 31 2c  ame) values ($1,
3610: 20 24 32 29 3b 22 29 0a 09 09 09 09 09 09 09 09   $2);").........
3620: 09 09 09 09 09 09 09 2e 62 69 6e 64 28 69 36 34  ........bind(i64
3630: 3a 3a 66 72 6f 6d 28 63 68 61 6e 5f 69 64 2e 69  ::from(chan_id.i
3640: 64 28 29 29 29 0a 09 09 09 09 09 09 09 09 09 09  d()))...........
3650: 09 09 09 09 09 2e 62 69 6e 64 28 63 68 61 6e 6e  ......bind(chann
3660: 65 6c 29 0a 09 09 09 09 09 09 09 09 09 09 09 09  el).............
3670: 09 09 09 2e 65 78 65 63 75 74 65 28 26 63 6f 72  ....execute(&cor
3680: 65 2e 70 6f 6f 6c 29 2e 61 77 61 69 74 3f 3b 0a  e.pool).await?;.
3690: 09 09 09 09 09 09 09 09 09 09 09 09 09 09 72 65  ..............re
36a0: 70 6c 79 2e 70 75 73 68 28 22 47 6f 6f 64 2c 20  ply.push("Good, 
36b0: 49 20 6b 6e 6f 77 20 74 68 61 74 20 63 68 61 6e  I know that chan
36c0: 6e 65 6c 20 6e 6f 77 5c 5c 2e 5c 6e 22 2e 74 6f  nel now\\.\n".to
36d0: 5f 73 74 72 69 6e 67 28 29 29 3b 0a 09 09 09 09  _string());.....
36e0: 09 09 09 09 09 09 09 09 09 7d 3b 0a 09 09 09 09  .........};.....
36f0: 09 09 09 09 09 09 09 09 7d 2c 0a 09 09 09 09 09  ........},......
3700: 09 09 09 09 09 09 09 45 72 72 28 5f 29 20 3d 3e  .......Err(_) =>
3710: 20 7b 0a 09 09 09 09 09 09 09 09 09 09 09 09 09   {..............
3720: 72 65 70 6c 79 2e 70 75 73 68 28 22 53 6f 72 72  reply.push("Sorr
3730: 79 2c 20 49 20 68 61 76 65 20 6e 6f 20 61 63 63  y, I have no acc
3740: 65 73 73 20 74 6f 20 74 68 61 74 20 63 68 61 74  ess to that chat
3750: 5c 5c 2e 22 2e 74 6f 5f 73 74 72 69 6e 67 28 29  \\.".to_string()
3760: 29 3b 0a 09 09 09 09 09 09 09 09 09 09 09 09 7d  );.............}
3770: 2c 0a 09 09 09 09 09 09 09 09 09 09 09 7d 3b 0a  ,............};.
3780: 09 09 09 09 09 09 09 09 09 09 7d 2c 0a 09 09 09  ..........},....
3790: 09 09 09 09 09 09 7d 3b 0a 09 09 09 09 09 09 09  ......};........
37a0: 09 7d 3b 0a 09 09 09 09 09 09 09 7d 2c 0a 0a 2f  .};........},../
37b0: 2f 20 63 68 65 63 6b 0a 0a 09 09 09 09 09 09 09  / check.........
37c0: 22 2f 63 68 65 63 6b 22 20 3d 3e 20 7b 0a 09 09  "/check" => {...
37d0: 09 09 09 09 09 09 6c 65 74 20 63 68 61 6e 6e 65  ......let channe
37e0: 6c 20 3d 20 77 6f 72 64 73 2e 6e 65 78 74 28 29  l = words.next()
37f0: 2e 75 6e 77 72 61 70 28 29 3b 0a 09 09 09 09 09  .unwrap();......
3800: 09 09 09 69 66 20 21 20 72 65 5f 75 73 65 72 6e  ...if ! re_usern
3810: 61 6d 65 2e 69 73 5f 6d 61 74 63 68 28 26 63 68  ame.is_match(&ch
3820: 61 6e 6e 65 6c 29 20 7b 0a 09 09 09 09 09 09 09  annel) {........
3830: 09 09 72 65 70 6c 79 2e 70 75 73 68 28 22 55 73  ..reply.push("Us
3840: 65 72 6e 61 6d 65 73 20 73 68 6f 75 6c 64 20 62  ernames should b
3850: 65 20 73 6f 6d 65 74 68 69 6e 67 20 6c 69 6b 65  e something like
3860: 20 5c 22 40 5c 5c 5b 61 2d 7a 5d 5c 5c 5b 61 2d   \"@\\[a-z]\\[a-
3870: 7a 30 2d 39 5f 5d 2b 5c 22 2c 20 61 72 65 6e 27  z0-9_]+\", aren'
3880: 74 20 74 68 65 79 3f 22 2e 74 6f 5f 73 74 72 69  t they?".to_stri
3890: 6e 67 28 29 29 3b 0a 09 09 09 09 09 09 09 09 7d  ng());.........}
38a0: 20 65 6c 73 65 20 7b 0a 09 09 09 09 09 09 09 09   else {.........
38b0: 09 26 63 6f 72 65 2e 63 68 65 63 6b 28 63 68 61  .&core.check(cha
38c0: 6e 6e 65 6c 2c 20 4e 6f 6e 65 29 2e 61 77 61 69  nnel, None).awai
38d0: 74 3f 3b 0a 09 09 09 09 09 09 09 09 7d 0a 09 09  t?;.........}...
38e0: 09 09 09 09 09 7d 2c 0a 0a 2f 2f 20 63 6c 65 61  .....},..// clea
38f0: 72 0a 0a 09 09 09 09 09 09 09 22 2f 63 6c 65 61  r........."/clea
3900: 6e 22 20 3d 3e 20 7b 0a 09 09 09 09 09 09 09 09  n" => {.........
3910: 69 66 20 63 6f 72 65 2e 6f 77 6e 65 72 20 21 3d  if core.owner !=
3920: 20 69 36 34 3a 3a 66 72 6f 6d 28 6d 65 73 73 61   i64::from(messa
3930: 67 65 2e 66 72 6f 6d 2e 69 64 29 20 7b 0a 09 09  ge.from.id) {...
3940: 09 09 09 09 09 09 09 72 65 70 6c 79 2e 70 75 73  .......reply.pus
3950: 68 28 22 52 65 73 65 72 76 65 64 20 66 6f 72 20  h("Reserved for 
3960: 74 65 73 74 69 6e 67 5c 5c 2e 22 2e 74 6f 5f 73  testing\\.".to_s
3970: 74 72 69 6e 67 28 29 29 3b 0a 09 09 09 09 09 09  tring());.......
3980: 09 09 7d 20 65 6c 73 65 20 7b 0a 09 09 09 09 09  ..} else {......
3990: 09 09 09 09 6c 65 74 20 73 6f 75 72 63 65 5f 69  ....let source_i
39a0: 64 20 3d 20 77 6f 72 64 73 2e 6e 65 78 74 28 29  d = words.next()
39b0: 2e 75 6e 77 72 61 70 28 29 2e 70 61 72 73 65 3a  .unwrap().parse:
39c0: 3a 3c 69 33 32 3e 28 29 2e 75 6e 77 72 61 70 5f  :<i32>().unwrap_
39d0: 6f 72 28 30 29 3b 0a 09 09 09 09 09 09 09 09 09  or(0);..........
39e0: 26 63 6f 72 65 2e 63 6c 65 61 6e 28 73 6f 75 72  &core.clean(sour
39f0: 63 65 5f 69 64 29 2e 61 77 61 69 74 3f 3b 0a 09  ce_id).await?;..
3a00: 09 09 09 09 09 09 09 7d 0a 09 09 09 09 09 09 09  .......}........
3a10: 7d 2c 0a 0a 2f 2f 20 65 6e 61 62 6c 65 0a 0a 09  },..// enable...
3a20: 09 09 09 09 09 09 22 2f 65 6e 61 62 6c 65 22 20  ......"/enable" 
3a30: 3d 3e 20 7b 0a 09 09 09 09 09 09 09 09 6c 65 74  => {.........let
3a40: 20 63 68 61 6e 6e 65 6c 20 3d 20 77 6f 72 64 73   channel = words
3a50: 2e 6e 65 78 74 28 29 2e 75 6e 77 72 61 70 28 29  .next().unwrap()
3a60: 3b 0a 09 09 09 09 09 09 09 09 69 66 20 21 20 72  ;.........if ! r
3a70: 65 5f 75 73 65 72 6e 61 6d 65 2e 69 73 5f 6d 61  e_username.is_ma
3a80: 74 63 68 28 26 63 68 61 6e 6e 65 6c 29 20 7b 0a  tch(&channel) {.
3a90: 09 09 09 09 09 09 09 09 09 72 65 70 6c 79 2e 70  .........reply.p
3aa0: 75 73 68 28 22 55 73 65 72 6e 61 6d 65 73 20 73  ush("Usernames s
3ab0: 68 6f 75 6c 64 20 62 65 20 73 6f 6d 65 74 68 69  hould be somethi
3ac0: 6e 67 20 6c 69 6b 65 20 5c 22 40 5c 5c 5b 61 2d  ng like \"@\\[a-
3ad0: 7a 5d 5c 5c 5b 61 2d 7a 30 2d 39 5f 5d 2b 5c 22  z]\\[a-z0-9_]+\"
3ae0: 2c 20 61 72 65 6e 27 74 20 74 68 65 79 3f 22 2e  , aren't they?".
3af0: 74 6f 5f 73 74 72 69 6e 67 28 29 29 3b 0a 09 09  to_string());...
3b00: 09 09 09 09 09 09 7d 20 65 6c 73 65 20 7b 0a 09  ......} else {..
3b10: 09 09 09 09 09 09 09 09 6d 61 74 63 68 20 63 6f  ........match co
3b20: 72 65 2e 65 6e 61 62 6c 65 28 6d 65 73 73 61 67  re.enable(messag
3b30: 65 2e 66 72 6f 6d 2e 69 64 2c 20 63 68 61 6e 6e  e.from.id, chann
3b40: 65 6c 29 2e 61 77 61 69 74 20 7b 0a 09 09 09 09  el).await {.....
3b50: 09 09 09 09 09 09 4f 6b 28 5f 29 20 3d 3e 20 7b  ......Ok(_) => {
3b60: 0a 09 09 09 09 09 09 09 09 09 09 09 72 65 70 6c  ............repl
3b70: 79 2e 70 75 73 68 28 22 43 68 61 6e 6e 65 6c 20  y.push("Channel 
3b80: 65 6e 61 62 6c 65 64 5c 5c 2e 22 2e 74 6f 5f 73  enabled\\.".to_s
3b90: 74 72 69 6e 67 28 29 29 3b 0a 09 09 09 09 09 09  tring());.......
3ba0: 09 09 09 09 7d 0a 09 09 09 09 09 09 09 09 09 09  ....}...........
3bb0: 45 72 72 28 65 72 72 29 20 3d 3e 20 7b 0a 09 09  Err(err) => {...
3bc0: 09 09 09 09 09 09 09 09 09 63 6f 72 65 2e 64 65  .........core.de
3bd0: 62 75 67 28 26 65 72 72 2e 74 6f 5f 73 74 72 69  bug(&err.to_stri
3be0: 6e 67 28 29 29 3f 3b 0a 09 09 09 09 09 09 09 09  ng())?;.........
3bf0: 09 09 7d 2c 0a 09 09 09 09 09 09 09 09 09 7d 0a  ..},..........}.
3c00: 09 09 09 09 09 09 09 09 7d 0a 09 09 09 09 09 09  ........}.......
3c10: 09 7d 2c 0a 0a 2f 2f 20 64 69 73 61 62 6c 65 0a  .},..// disable.
3c20: 0a 09 09 09 09 09 09 09 22 2f 64 69 73 61 62 6c  ........"/disabl
3c30: 65 22 20 3d 3e 20 7b 0a 09 09 09 09 09 09 09 09  e" => {.........
3c40: 6c 65 74 20 63 68 61 6e 6e 65 6c 20 3d 20 77 6f  let channel = wo
3c50: 72 64 73 2e 6e 65 78 74 28 29 2e 75 6e 77 72 61  rds.next().unwra
3c60: 70 28 29 3b 0a 09 09 09 09 09 09 09 09 69 66 20  p();.........if 
3c70: 21 20 72 65 5f 75 73 65 72 6e 61 6d 65 2e 69 73  ! re_username.is
3c80: 5f 6d 61 74 63 68 28 26 63 68 61 6e 6e 65 6c 29  _match(&channel)
3c90: 20 7b 0a 09 09 09 09 09 09 09 09 09 72 65 70 6c   {..........repl
3ca0: 79 2e 70 75 73 68 28 22 55 73 65 72 6e 61 6d 65  y.push("Username
3cb0: 73 20 73 68 6f 75 6c 64 20 62 65 20 73 6f 6d 65  s should be some
3cc0: 74 68 69 6e 67 20 6c 69 6b 65 20 5c 22 40 5c 5c  thing like \"@\\
3cd0: 5b 61 2d 7a 5d 5c 5c 5b 61 2d 7a 30 2d 39 5f 5d  [a-z]\\[a-z0-9_]
3ce0: 2b 5c 22 2c 20 61 72 65 6e 27 74 20 74 68 65 79  +\", aren't they
3cf0: 3f 22 2e 74 6f 5f 73 74 72 69 6e 67 28 29 29 3b  ?".to_string());
3d00: 0a 09 09 09 09 09 09 09 09 7d 20 65 6c 73 65 20  .........} else 
3d10: 7b 0a 09 09 09 09 09 09 09 09 09 6d 61 74 63 68  {..........match
3d20: 20 63 6f 72 65 2e 64 69 73 61 62 6c 65 28 6d 65   core.disable(me
3d30: 73 73 61 67 65 2e 66 72 6f 6d 2e 69 64 2c 20 63  ssage.from.id, c
3d40: 68 61 6e 6e 65 6c 29 2e 61 77 61 69 74 20 7b 0a  hannel).await {.
3d50: 09 09 09 09 09 09 09 09 09 09 4f 6b 28 5f 29 20  ..........Ok(_) 
3d60: 3d 3e 20 7b 0a 09 09 09 09 09 09 09 09 09 09 09  => {............
3d70: 72 65 70 6c 79 2e 70 75 73 68 28 22 43 68 61 6e  reply.push("Chan
3d80: 6e 65 6c 20 64 69 73 61 62 6c 65 64 5c 5c 2e 22  nel disabled\\."
3d90: 2e 74 6f 5f 73 74 72 69 6e 67 28 29 29 3b 0a 09  .to_string());..
3da0: 09 09 09 09 09 09 09 09 09 7d 0a 09 09 09 09 09  .........}......
3db0: 09 09 09 09 09 45 72 72 28 65 72 72 29 20 3d 3e  .....Err(err) =>
3dc0: 20 7b 0a 09 09 09 09 09 09 09 09 09 09 09 63 6f   {............co
3dd0: 72 65 2e 64 65 62 75 67 28 26 65 72 72 2e 74 6f  re.debug(&err.to
3de0: 5f 73 74 72 69 6e 67 28 29 29 3f 3b 0a 09 09 09  _string())?;....
3df0: 09 09 09 09 09 09 09 7d 2c 0a 09 09 09 09 09 09  .......},.......
3e00: 09 09 09 7d 0a 09 09 09 09 09 09 09 09 7d 0a 09  ...}.........}..
3e10: 09 09 09 09 09 09 7d 2c 0a 0a 09 09 09 09 09 09  ......},........
3e20: 09 5f 20 3d 3e 20 7b 0a 09 09 09 09 09 09 09 7d  ._ => {........}
3e30: 2c 0a 09 09 09 09 09 09 7d 3b 0a 09 09 09 09 09  ,.......};......
3e40: 7d 2c 0a 09 09 09 09 09 5f 20 3d 3e 20 7b 0a 09  },......_ => {..
3e50: 09 09 09 09 7d 2c 0a 09 09 09 09 7d 3b 0a 09 09  ....},.....};...
3e60: 09 09 69 66 20 72 65 70 6c 79 2e 6c 65 6e 28 29  ..if reply.len()
3e70: 20 3e 20 30 20 7b 0a 09 09 09 09 09 6d 61 74 63   > 0 {......matc
3e80: 68 20 63 6f 72 65 2e 74 67 2e 73 65 6e 64 28 6d  h core.tg.send(m
3e90: 65 73 73 61 67 65 2e 74 65 78 74 5f 72 65 70 6c  essage.text_repl
3ea0: 79 28 72 65 70 6c 79 2e 6a 6f 69 6e 28 22 5c 6e  y(reply.join("\n
3eb0: 22 29 29 2e 70 61 72 73 65 5f 6d 6f 64 65 28 74  ")).parse_mode(t
3ec0: 79 70 65 73 3a 3a 50 61 72 73 65 4d 6f 64 65 3a  ypes::ParseMode:
3ed0: 3a 4d 61 72 6b 64 6f 77 6e 56 32 29 29 2e 61 77  :MarkdownV2)).aw
3ee0: 61 69 74 20 7b 0a 09 09 09 09 09 09 4f 6b 28 5f  ait {.......Ok(_
3ef0: 29 20 3d 3e 20 7b 7d 2c 0a 09 09 09 09 09 09 45  ) => {},.......E
3f00: 72 72 28 65 72 72 29 20 3d 3e 20 7b 0a 09 09 09  rr(err) => {....
3f10: 09 09 09 09 64 62 67 21 28 72 65 70 6c 79 2e 6a  ....dbg!(reply.j
3f20: 6f 69 6e 28 22 5c 6e 22 29 29 3b 0a 09 09 09 09  oin("\n"));.....
3f30: 09 09 09 70 72 69 6e 74 6c 6e 21 28 22 7b 7d 22  ...println!("{}"
3f40: 2c 20 65 72 72 29 3b 0a 09 09 09 09 09 09 7d 2c  , err);.......},
3f50: 0a 09 09 09 09 09 7d 0a 09 09 09 09 7d 0a 09 09  ......}.....}...
3f60: 09 7d 2c 0a 09 09 09 5f 20 3d 3e 20 7b 7d 2c 0a  .},...._ => {},.
3f70: 09 09 7d 3b 0a 09 7d 0a 0a 09 4f 6b 28 28 29 29  ..};..}...Ok(())
3f80: 0a 7d 0a                                         .}.