diff options
author | jan <jan@ruken.pw> | 2016-04-07 16:16:03 (UTC) |
---|---|---|
committer | jan <jan@ruken.pw> | 2016-04-07 16:16:03 (UTC) |
commit | 609fa1ab6351dd58a8c9cd245aeec0eaf647c9e9 (patch) | |
tree | 1ba75752cccb7a61ef32263ccbffc7e6fd46ce43 /src | |
parent | 47cf2e6ee0489d2504a61e39e76dd88c5747694d (diff) |
feature TopAnime hinzugefuegt, wird automatisch geupdated
Diffstat (limited to 'src')
-rw-r--r-- | src/feature/mod.rs | 6 | ||||
-rw-r--r-- | src/feature/topanime.rs | 119 | ||||
-rw-r--r-- | src/main.rs | 2 |
3 files changed, 126 insertions, 1 deletions
diff --git a/src/feature/mod.rs b/src/feature/mod.rs index a52bb2c..b57c179 100644 --- a/src/feature/mod.rs +++ b/src/feature/mod.rs | |||
@@ -2,9 +2,11 @@ use std::boxed::Box; | |||
2 | 2 | ||
3 | extern crate telegram_bot; | 3 | extern crate telegram_bot; |
4 | use telegram_bot::{Api, Message}; | 4 | use telegram_bot::{Api, Message}; |
5 | extern crate rustc_serialize; | ||
5 | 6 | ||
6 | pub mod jisoku; | 7 | pub mod jisoku; |
7 | pub mod tasterank; | 8 | pub mod tasterank; |
9 | pub mod topanime; | ||
8 | 10 | ||
9 | pub enum FeatureResult { | 11 | pub enum FeatureResult { |
10 | Handled, | 12 | Handled, |
@@ -18,5 +20,7 @@ pub trait Feature { | |||
18 | } | 20 | } |
19 | 21 | ||
20 | pub fn init() -> Vec<Box<Feature>> { | 22 | pub fn init() -> Vec<Box<Feature>> { |
21 | vec![Box::new(tasterank::Tasterank::new()), Box::new(jisoku::Jisoku::new())] | 23 | vec![Box::new(tasterank::Tasterank::new()), |
24 | Box::new(jisoku::Jisoku::new()), | ||
25 | Box::new(topanime::TopAnime::new())] | ||
22 | } | 26 | } |
diff --git a/src/feature/topanime.rs b/src/feature/topanime.rs new file mode 100644 index 0000000..cccc976 --- /dev/null +++ b/src/feature/topanime.rs | |||
@@ -0,0 +1,119 @@ | |||
1 | use std::io; | ||
2 | use std::io::prelude::*; | ||
3 | use std::fs::File; | ||
4 | use std::usize; | ||
5 | |||
6 | extern crate telegram_bot; | ||
7 | use telegram_bot::{Api, Message, MessageType}; | ||
8 | extern crate time; | ||
9 | extern crate regex; | ||
10 | use self::regex::Regex; | ||
11 | use super::rustc_serialize::json; | ||
12 | |||
13 | use feature::FeatureResult; | ||
14 | use feature::Feature; | ||
15 | |||
16 | #[derive(RustcDecodable)] | ||
17 | pub struct Anime { | ||
18 | name: String, | ||
19 | score: f64, | ||
20 | } | ||
21 | |||
22 | impl Anime { | ||
23 | pub fn new(name: &str, score: f64) -> Anime { | ||
24 | Anime { | ||
25 | name: name.to_owned(), | ||
26 | score: score, | ||
27 | } | ||
28 | } | ||
29 | } | ||
30 | |||
31 | pub struct TopAnime { | ||
32 | animes: Vec<Anime>, | ||
33 | last_access: f64, | ||
34 | } | ||
35 | |||
36 | impl Feature for TopAnime { | ||
37 | fn name(&self) -> &'static str { | ||
38 | "Tasterank" | ||
39 | } | ||
40 | fn init(&mut self) {} | ||
41 | fn handle(&mut self, a: Api, m: Message) -> Result<FeatureResult, String> { | ||
42 | if let MessageType::Text(s) = m.msg { | ||
43 | let re = Regex::new(r"(?i)luggas top ([0-9]{1,3}) anime").unwrap(); | ||
44 | let cap = re.captures(&s); | ||
45 | let s_num = match cap { | ||
46 | Some(s) => s.at(1), | ||
47 | None => return Ok(FeatureResult::Skip), | ||
48 | }; | ||
49 | let s_num = match s_num { | ||
50 | Some(s) => s, | ||
51 | None => return Ok(FeatureResult::Skip), | ||
52 | }; | ||
53 | |||
54 | // parse to int | ||
55 | let num = match s_num.parse::<i32>() { | ||
56 | Ok(n) => n, | ||
57 | Err(e) => { | ||
58 | println!("{}", e); | ||
59 | return Ok(FeatureResult::Skip); | ||
60 | } | ||
61 | }; | ||
62 | let start = time::precise_time_ns(); | ||
63 | if let Err(e) = self.update_animes() { | ||
64 | println!("{}", e); | ||
65 | return Ok(FeatureResult::Skip); | ||
66 | } | ||
67 | println!("(benchm) updating animes: {}ms", | ||
68 | (time::precise_time_ns() - start) / 1000000); | ||
69 | |||
70 | let mut msg = String::new(); | ||
71 | for i in 0..num { | ||
72 | let i = i as usize; | ||
73 | if self.animes.len() <= i { | ||
74 | break; | ||
75 | } | ||
76 | msg.push_str(&format!("{}. {} ({}%)\n", | ||
77 | i + 1, | ||
78 | self.animes[i].name, | ||
79 | self.animes[i].score)); | ||
80 | } | ||
81 | println!("(benchm) anime before send: {}ms", | ||
82 | (time::precise_time_ns() - start) / 1000000); | ||
83 | if let Err(e) = a.send_message(m.chat.id(), msg, None, None, Some(m.message_id), None) { | ||
84 | println!("{}", e); | ||
85 | } | ||
86 | println!("(benchm) anime: {}ms", | ||
87 | (time::precise_time_ns() - start) / 1000000); | ||
88 | return Ok(FeatureResult::Handled); | ||
89 | } | ||
90 | Ok(FeatureResult::Skip) | ||
91 | } | ||
92 | } | ||
93 | |||
94 | impl TopAnime { | ||
95 | pub fn new() -> TopAnime { | ||
96 | let mut t = TopAnime { | ||
97 | animes: vec![], | ||
98 | last_access: 0f64, | ||
99 | }; | ||
100 | t.init(); | ||
101 | t | ||
102 | } | ||
103 | fn update_animes(&mut self) -> Result<(), io::Error> { | ||
104 | let now = time::precise_time_s(); | ||
105 | if now - self.last_access < 3600.0f64 { | ||
106 | return Ok(()); | ||
107 | } | ||
108 | let mut f = try!(File::open("data/animescore.json")); | ||
109 | let mut buf = String::new(); | ||
110 | try!(f.read_to_string(&mut buf)); | ||
111 | let v: Vec<Anime> = match json::decode(&buf) { | ||
112 | Ok(v) => v, | ||
113 | Err(e) => return Err(io::Error::new(io::ErrorKind::InvalidData, format!("{}", e))), | ||
114 | }; | ||
115 | self.animes = v; | ||
116 | self.last_access = now; | ||
117 | Ok(()) | ||
118 | } | ||
119 | } | ||
diff --git a/src/main.rs b/src/main.rs index ef5c8e8..0880467 100644 --- a/src/main.rs +++ b/src/main.rs | |||
@@ -1,5 +1,7 @@ | |||
1 | extern crate telegram_bot; | 1 | extern crate telegram_bot; |
2 | use telegram_bot::{Api, ListeningMethod, ListeningAction}; | 2 | use telegram_bot::{Api, ListeningMethod, ListeningAction}; |
3 | extern crate rustc_serialize; | ||
4 | use rustc_serialize::json; | ||
3 | 5 | ||
4 | mod feature; | 6 | mod feature; |
5 | use feature::{Feature, FeatureResult}; | 7 | use feature::{Feature, FeatureResult}; |