def test_run__sleeps_backoff_time(self, bot): api = MockExportAPI() watcher = SubscriptionWatcher(api, bot) # Shorten the wait watcher.BACK_OFF = 3 api.call_after_x_browse = (lambda: watcher.stop(), 2) # Run watcher start_time = datetime.datetime.now() watcher.run() end_time = datetime.datetime.now() time_waited = end_time - start_time assert 3 <= time_waited.seconds <= 5
def test_run__passes_correct_blocklists_to_subscriptions(self, bot): submission = MockSubmission("12322") api = MockExportAPI().with_submission(submission) watcher = SubscriptionWatcher(api, bot) method_called = MockMethod([submission]) watcher._get_new_results = method_called.call watcher.BACK_OFF = 1 watcher.blocklists = {156: {"test", "ych"}, -200: {"example"}} sub1 = MockSubscription("deer", 156) sub2 = MockSubscription("dog", -232) watcher.subscriptions = [sub1, sub2] thread = Thread(target=lambda: self.watcher_killer(watcher)) thread.start() # Run watcher watcher.run() thread.join() assert submission in sub1.submissions_checked assert len(sub1.blocklists) == 1 assert len(sub1.blocklists[0]) == 2 assert "test" in sub1.blocklists[0] assert "ych" in sub1.blocklists[0] assert submission in sub2.submissions_checked assert len(sub2.blocklists) == 1 assert len(sub2.blocklists[0]) == 0 assert method_called.called
def test_user_favourites(context): post_id1 = 234563 post_id2 = 393282 username = "******" update = MockTelegramUpdate.with_inline_query( query=f"favourites:{username}") submission1 = MockSubmission(post_id1) submission2 = MockSubmission(post_id2) inline = InlineFunctionality(MockExportAPI()) inline.api.with_user_favs(username, [submission1, submission2]) inline.call(update, context) context.bot.answer_inline_query.assert_called_once() args = context.bot.answer_inline_query.call_args[0] assert context.bot.answer_inline_query.call_args[1][ 'next_offset'] == submission2.fav_id assert args[0] == update.inline_query.id assert isinstance(args[1], list) assert len(args[1]) == 2 assert isinstance(args[1][0], InlineQueryResultPhoto) assert isinstance(args[1][1], InlineQueryResultPhoto) assert args[1][0].id == str(post_id1) assert args[1][1].id == str(post_id2) assert args[1][0].photo_url == submission1.thumbnail_url assert args[1][1].photo_url == submission2.thumbnail_url assert args[1][0].thumb_url == FASubmission.make_thumbnail_smaller( submission1.thumbnail_url) assert args[1][1].thumb_url == FASubmission.make_thumbnail_smaller( submission2.thumbnail_url) assert args[1][0].caption == submission1.link assert args[1][1].caption == submission2.link
def test_get_new_results__returns_new_results(self, bot): api = MockExportAPI() api.with_browse_results([ MockSubmission("1223"), MockSubmission("1222"), MockSubmission("1220") ]) watcher = SubscriptionWatcher(api, bot) watcher.latest_ids.append("1220") watcher.running = True results = watcher._get_new_results() assert len(results) == 2 assert results[0].submission_id == "1222" assert results[1].submission_id == "1223"
def test_direct_no_match_groupchat(context): username = "******" image_id = 1560331512 post_id = 232347 update = MockTelegramUpdate.with_message( text="http://d.facdn.net/art/{0}/{1}/{1}.pic_of_me.png".format(username, image_id), chat_type=Chat.GROUP ) neaten = NeatenFunctionality(MockExportAPI()) for folder in ['gallery', 'scraps']: neaten.api.with_user_folder(username, folder, [ MockSubmission(post_id, image_id=image_id + 4), MockSubmission(post_id - 1, image_id=image_id - 15) ]) neaten.call(update, context) context.bot.send_photo.assert_not_called() context.bot.send_message.assert_called_with( chat_id=update.message.chat_id, text="⏳ Neatening image link", reply_to_message_id=update.message.message_id ) context.bot.delete_message.assert_called_with( update.message.chat_id, context._sent_message_ids[0] )
def test_no_username_set(context, requests_mock): username = "" update = MockTelegramUpdate.with_inline_query(query=f"favs:{username}") inline = InlineFunctionality(MockExportAPI()) # mock export api doesn't do non-existent users, so mocking with requests inline.api = FAExportAPI("http://example.com") requests_mock.get( f"http://example.com/user/{username}/favorites.json?page=1&full=1", json={ "id": None, "name": "favorites", "profile": "https://www.furaffinity.net/user/favorites/" }) inline.call(update, context) context.bot.answer_inline_query.assert_called_once() args = context.bot.answer_inline_query.call_args[0] assert context.bot.answer_inline_query.call_args[1]['next_offset'] == "" assert args[0] == update.inline_query.id assert isinstance(args[1], list) assert len(args[1]) == 1 assert isinstance(args[1][0], InlineQueryResultArticle) assert args[1][0].title == "User does not exist." assert isinstance(args[1][0].input_message_content, InputMessageContent) assert args[1][0].input_message_content.message_text == \ f"FurAffinity user does not exist by the name: \"{username}\"."
def test_submission_link_and_different_direct_link(context): username = "******" image_id1 = 1560331512 image_id2 = image_id1 + 300 post_id1 = 232347 post_id2 = 233447 update = MockTelegramUpdate.with_message( text="https://furaffinity.net/view/{2}/ http://d.facdn.net/art/{0}/{1}/{1}.pic_of_me.png".format( username, image_id1, post_id2 ) ) submission1 = MockSubmission(post_id1, image_id=image_id1) submission2 = MockSubmission(post_id2, image_id=image_id2) neaten = NeatenFunctionality(MockExportAPI()) neaten.api.with_user_folder(username, "gallery", [ submission2, submission1, MockSubmission(post_id1 - 1, image_id=image_id1 - 15) ]) neaten.call(update, context) context.bot.send_photo.assert_called() calls = [call( chat_id=update.message.chat_id, photo=submission.download_url, caption=submission.link, reply_to_message_id=update.message.message_id ) for submission in [submission2, submission1]] context.bot.send_photo.assert_has_calls(calls)
def test_over_48_favs(context): username = "******" post_ids = list(range(123456, 123456 + 72)) submissions = [MockSubmission(x) for x in post_ids] inline = InlineFunctionality(MockExportAPI()) inline.api.with_user_favs(username, submissions) update = MockTelegramUpdate.with_inline_query(query=f"favs:{username}") inline.call(update, context) context.bot.answer_inline_query.assert_called_once() args = context.bot.answer_inline_query.call_args[0] assert context.bot.answer_inline_query.call_args[1][ 'next_offset'] == submissions[47].fav_id assert args[0] == update.inline_query.id assert isinstance(args[1], list) assert len(args[1]) == 48 assert isinstance(args[1][0], InlineQueryResultPhoto) assert isinstance(args[1][1], InlineQueryResultPhoto) for x in range(48): assert args[1][x].id == str(post_ids[x]) assert args[1][x].photo_url == submissions[x].thumbnail_url assert args[1][x].thumb_url == FASubmission.make_thumbnail_smaller( submissions[x].thumbnail_url) assert args[1][x].caption == submissions[x].link
def test_username_with_colon(context, requests_mock): # FA doesn't allow usernames to have : in them username = "******" update = MockTelegramUpdate.with_inline_query(query=f"gallery:{username}") inline = InlineFunctionality(MockExportAPI()) # mock export api doesn't do non-existent users, so mocking with requests inline.api = FAExportAPI("http://example.com") requests_mock.get( f"http://example.com/user/{username}/gallery.json", status_code=404 ) inline.call(update, context) context.bot.answer_inline_query.assert_called_once() args = context.bot.answer_inline_query.call_args[0] assert context.bot.answer_inline_query.call_args[1]['next_offset'] == "" assert args[0] == update.inline_query.id assert isinstance(args[1], list) assert len(args[1]) == 1 assert isinstance(args[1][0], InlineQueryResultArticle) assert args[1][0].title == "User does not exist." assert isinstance(args[1][0].input_message_content, InputMessageContent) assert args[1][0].input_message_content.message_text == \ f"FurAffinity user does not exist by the name: \"{username}\"."
def test_search_with_spaces(context): search_term = "deer YCH" update = MockTelegramUpdate.with_inline_query(query=search_term) post_id1 = 213231 post_id2 = 84331 submission1 = MockSubmission(post_id1) submission2 = MockSubmission(post_id2) inline = InlineFunctionality(MockExportAPI()) inline.api.with_search_results(search_term, [submission1, submission2]) inline.call(update, context) context.bot.answer_inline_query.assert_called_once() args = context.bot.answer_inline_query.call_args[0] assert context.bot.answer_inline_query.call_args[1]['next_offset'] == 2 assert args[0] == update.inline_query.id assert isinstance(args[1], list) assert len(args[1]) == 2 for result in args[1]: assert isinstance(result, InlineQueryResultPhoto) assert args[1][0].id == str(post_id1) assert args[1][1].id == str(post_id2) assert args[1][0].photo_url == submission1.thumbnail_url assert args[1][1].photo_url == submission2.thumbnail_url assert args[1][0].thumb_url == FASubmission.make_thumbnail_smaller( submission1.thumbnail_url) assert args[1][1].thumb_url == FASubmission.make_thumbnail_smaller( submission2.thumbnail_url) assert args[1][0].caption == submission1.link assert args[1][1].caption == submission2.link
def test_result_first_on_page(context): username = "******" image_id = 1560331512 post_id = 232347 update = MockTelegramUpdate.with_message( text="http://d.facdn.net/art/{0}/{1}/{1}.pic_of_me.png".format(username, image_id) ) submission = MockSubmission(post_id, image_id=image_id) neaten = NeatenFunctionality(MockExportAPI()) neaten.api.with_user_folder(username, "gallery", [ MockSubmission(post_id + 3, image_id=image_id + 16), MockSubmission(post_id + 2, image_id=image_id + 8) ], page=1) neaten.api.with_user_folder(username, "gallery", [ submission, MockSubmission(post_id - 2, image_id=image_id - 2), MockSubmission(post_id - 7, image_id=image_id - 4), MockSubmission(post_id - 9, image_id=image_id - 10) ], page=2) neaten.call(update, context) context.bot.send_photo.assert_called_once() assert context.bot.send_photo.call_args[1]['chat_id'] == update.message.chat_id assert context.bot.send_photo.call_args[1]['photo'] == submission.download_url assert context.bot.send_photo.call_args[1]['caption'] == submission.link assert context.bot.send_photo.call_args[1]['reply_to_message_id'] == update.message.message_id
def test_result_missing_between_pages(context): username = "******" image_id = 1560331512 post_id = 232347 update = MockTelegramUpdate.with_message( text="http://d.facdn.net/art/{0}/{1}/{1}.pic_of_me.png".format(username, image_id) ) neaten = NeatenFunctionality(MockExportAPI()) neaten.api.with_user_folder(username, "gallery", [ MockSubmission(post_id + 1, image_id=image_id + 16), MockSubmission(post_id, image_id=image_id + 3) ], page=1) neaten.api.with_user_folder(username, "gallery", [ MockSubmission(post_id - 2, image_id=image_id - 27), MockSubmission(post_id - 3, image_id=image_id - 34) ], page=2) neaten.call(update, context) context.bot.send_photo.assert_not_called() context.bot.send_message.assert_called_with( chat_id=update.message.chat_id, text="Could not locate the image by {} with image id {}.".format(username, image_id), reply_to_message_id=update.message.message_id )
def test_result_in_scraps(context): username = "******" image_id = 1560331512 post_id = 232347 update = MockTelegramUpdate.with_message( text="http://d.facdn.net/art/{0}/{1}/{1}.pic_of_me.png".format(username, image_id) ) submission = MockSubmission(post_id, image_id=image_id) neaten = NeatenFunctionality(MockExportAPI()) for page in [1, 2]: neaten.api.with_user_folder(username, "gallery", [ MockSubmission(post_id + 1 + (3 - page) * 5, image_id=image_id + 16 + (3 - page) * 56), MockSubmission(post_id + (3 - page) * 5, image_id=image_id + (3 - page) * 56), MockSubmission(post_id - 2 + (3 - page) * 5, image_id=image_id - 27 + (3 - page) * 56), MockSubmission(post_id - 3 + (3 - page) * 5, image_id=image_id - 34 + (3 - page) * 56) ], page=page) neaten.api.with_user_folder(username, "gallery", [], page=3) neaten.api.with_user_folder(username, "scraps", [ MockSubmission(post_id + 1, image_id=image_id + 16), submission, MockSubmission(post_id - 2, image_id=image_id - 27), MockSubmission(post_id - 3, image_id=image_id - 34) ], page=1) neaten.call(update, context) context.bot.send_photo.assert_called_once() assert context.bot.send_photo.call_args[1]['chat_id'] == update.message.chat_id assert context.bot.send_photo.call_args[1]['photo'] == submission.download_url assert context.bot.send_photo.call_args[1]['caption'] == submission.link assert context.bot.send_photo.call_args[1]['reply_to_message_id'] == update.message.message_id
def test_direct_in_progress_message(context): username = "******" image_id = 1560331512 post_id = 232347 update = MockTelegramUpdate.with_message( text="http://d.facdn.net/art/{0}/{1}/{1}.pic_of_me.png".format(username, image_id) ) goal_submission = MockSubmission(post_id, image_id=image_id) neaten = NeatenFunctionality(MockExportAPI()) neaten.api.with_user_folder(username, "gallery", [ goal_submission, MockSubmission(post_id - 1, image_id=image_id - 15) ]) neaten.call(update, context) context.bot.send_message.assert_called_with( chat_id=update.message.chat_id, text="⏳ Neatening image link", reply_to_message_id=update.message.message_id ) context.bot.delete_message.assert_called_with( update.message.chat_id, context._sent_message_ids[0] )
def test_remove_sub__removes_subscription(context): api = MockExportAPI() watcher = SubscriptionWatcher(api, context.bot) watcher.subscriptions.add(Subscription("example", 18749)) watcher.subscriptions.add(Subscription("test", 18747)) new_sub = Subscription("test", 18749) new_sub.latest_update = datetime.datetime.now() watcher.subscriptions.add(new_sub) func = SubscriptionFunctionality(watcher) list_subs = MockMethod("Listing subscriptions") func._list_subs = list_subs.call resp = func._remove_sub(18749, "test") assert "Removed subscription: \"test\"." in resp assert list_subs.called assert list_subs.args[0] == 18749 assert "Listing subscriptions" in resp assert len(watcher.subscriptions) == 2 subscriptions = list(watcher.subscriptions) if subscriptions[0].query == "test": assert subscriptions[0].destination == 18747 assert subscriptions[1].query == "example" assert subscriptions[1].destination == 18749 else: assert subscriptions[0].query == "example" assert subscriptions[0].destination == 18749 assert subscriptions[1].query == "test" assert subscriptions[1].destination == 18747
def test_continue_from_fav_id(context): post_id = 234563 fav_id = "354233" username = "******" update = MockTelegramUpdate.with_inline_query(query=f"favs:{username}", offset=fav_id) submission = MockSubmission(post_id) inline = InlineFunctionality(MockExportAPI()) inline.api.with_user_favs(username, [submission], next_id=fav_id) inline.call(update, context) context.bot.answer_inline_query.assert_called_once() args = context.bot.answer_inline_query.call_args[0] assert context.bot.answer_inline_query.call_args[1][ 'next_offset'] == submission.fav_id assert args[0] == update.inline_query.id assert isinstance(args[1], list) assert len(args[1]) == 1 assert isinstance(args[1][0], InlineQueryResultPhoto) assert args[1][0].id == str(post_id) assert args[1][0].photo_url == submission.thumbnail_url assert args[1][0].thumb_url == FASubmission.make_thumbnail_smaller( submission.thumbnail_url) assert args[1][0].caption == submission.link
def test_run__failed_to_send_doesnt_kill_watcher(self, bot): submission = MockSubmission("12322") api = MockExportAPI().with_browse_results([submission], 1) watcher = SubscriptionWatcher(api, bot) watcher._send_update = lambda *args: (_ for _ in ()).throw(Exception) watcher.BACK_OFF = 3 sub1 = MockSubscription("deer", 0) watcher.subscriptions = [sub1] api.call_after_x_browse = (lambda: watcher.stop(), 2) # Run watcher start_time = datetime.datetime.now() watcher.run() end_time = datetime.datetime.now() time_waited = end_time - start_time assert 3 <= time_waited.seconds <= 5
def test_init(self, bot): api = MockExportAPI() s = SubscriptionWatcher(api, bot) assert s.api == api assert len(s.latest_ids) == 0 assert s.running is False assert len(s.subscriptions) == 0
def test_ignore_message(context): update = MockTelegramUpdate.with_message(text="hello world") neaten = NeatenFunctionality(MockExportAPI()) neaten.call(update, context) context.bot.send_message.assert_not_called() context.bot.send_photo.assert_not_called()
def test_ignore_journal_link(context): update = MockTelegramUpdate.with_message(text="https://www.furaffinity.net/journal/9150534/") neaten = NeatenFunctionality(MockExportAPI()) neaten.call(update, context) context.bot.send_message.assert_not_called() context.bot.send_photo.assert_not_called()
def test_get_new_results__handles_empty_latest_ids(self, bot): api = MockExportAPI() api.with_browse_results([ MockSubmission("1223"), MockSubmission("1222"), MockSubmission("1220") ]) watcher = SubscriptionWatcher(api, bot) watcher.running = True results = watcher._get_new_results() assert len(results) == 0 assert len(watcher.latest_ids) == 3 assert watcher.latest_ids[0] == "1220" assert watcher.latest_ids[1] == "1222" assert watcher.latest_ids[2] == "1223"
def test_ignore_link(context): update = MockTelegramUpdate.with_message(text="http://example.com") neaten = NeatenFunctionality(MockExportAPI()) neaten.call(update, context) context.bot.send_message.assert_not_called() context.bot.send_photo.assert_not_called()
def test_get_new_results__respects_page_cap(self, bot): api = MockExportAPI() api.with_browse_results([MockSubmission("1300")], page=1) api.with_browse_results([MockSubmission("1298")], page=2) api.with_browse_results([MockSubmission("1297")], page=3) api.with_browse_results([MockSubmission("1295")], page=4) api.with_browse_results([MockSubmission("1280")], page=5) api.with_browse_results([MockSubmission("1272")], page=6) api.with_browse_results([MockSubmission("1250")], page=7) watcher = SubscriptionWatcher(api, bot) watcher.PAGE_CAP = 5 watcher.latest_ids.append("1250") watcher.running = True results = watcher._get_new_results() assert len(results) == 5 assert "1272" not in [x.submission_id for x in results]
def test_send_update__updates_latest(self, bot): api = MockExportAPI() watcher = SubscriptionWatcher(api, bot) subscription = Subscription("test", 12345) submission = SubmissionBuilder().build_mock_submission() watcher._send_update(subscription, submission) assert subscription.latest_update is not None
def test_add_to_blocklist__new_blocklist(self, bot): api = MockExportAPI() watcher = SubscriptionWatcher(api, bot) watcher.add_to_blocklist(18749, "test") assert len(watcher.blocklists[18749]) == 1 assert isinstance(watcher.blocklists[18749], set) assert watcher.blocklists[18749] == {"test"}
def test_add_sub__no_add_blank(context): api = MockExportAPI() watcher = SubscriptionWatcher(api, context.bot) func = SubscriptionFunctionality(watcher) resp = func._add_sub(18749, "") assert resp == "Please specify the subscription query you wish to add." assert len(watcher.subscriptions) == 0
def test_add_to_blocklist__no_add_blank(context): api = MockExportAPI() watcher = SubscriptionWatcher(api, context.bot) func = BlocklistFunctionality(watcher) resp = func._add_to_blocklist(18749, "") assert resp == "Please specify the tag you wish to add to blocklist." assert len(watcher.blocklists) == 0
def test_empty_query_no_results(context): update = MockTelegramUpdate.with_inline_query(query="") inline = InlineFunctionality(MockExportAPI()) inline.call(update, context) context.bot.send_message.assert_not_called() context.bot.send_photo.assert_not_called() context.bot.answer_inline_query.assert_called_with(update.inline_query.id, [])
def test_doesnt_fire_on_avatar(context): update = MockTelegramUpdate.with_message( text="https://a.facdn.net/1538326752/geordie79.gif") neaten = NeatenFunctionality(MockExportAPI()) neaten.call(update, context) context.bot.send_photo.assert_not_called() context.bot.send_document.assert_not_called() context.bot.send_message.assert_not_called() context.bot.send_audio.assert_not_called()
def test_remove_sub__non_existent_subscription(context): api = MockExportAPI() watcher = SubscriptionWatcher(api, context.bot) watcher.subscriptions.add(Subscription("example", 18749)) watcher.subscriptions.add(Subscription("test", 18747)) func = SubscriptionFunctionality(watcher) resp = func._remove_sub(18749, "test") assert resp == "There is not a subscription for \"test\" in this chat." assert len(watcher.subscriptions) == 2