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__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_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_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_get_new_results__goes_to_another_page(self, bot): api = MockExportAPI() api.with_browse_results([ MockSubmission("1227"), MockSubmission("1225"), MockSubmission("1224") ], page=1) api.with_browse_results([ MockSubmission("1223"), MockSubmission("1222"), MockSubmission("1220") ], page=2) watcher = SubscriptionWatcher(api, bot) watcher.latest_ids = collections.deque(maxlen=4) watcher.latest_ids.append("1220") watcher.running = True results = watcher._get_new_results() assert len(results) == 5 assert results[0].submission_id == "1222" assert results[1].submission_id == "1223" assert results[2].submission_id == "1224" assert results[3].submission_id == "1225" assert results[4].submission_id == "1227"
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_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_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_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_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_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_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_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_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_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_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_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_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_to_json_and_back(self, bot): test_watcher_file = "./test_subscription_watcher.json" if os.path.exists(test_watcher_file): os.remove(test_watcher_file) old_filename = SubscriptionWatcher.FILENAME SubscriptionWatcher.FILENAME = test_watcher_file api = MockExportAPI() latest_submissions = [ SubmissionBuilder(submission_id="123243").build_short_submission(), SubmissionBuilder(submission_id="123242").build_short_submission(), SubmissionBuilder(submission_id="123240").build_short_submission() ] subscription1 = Subscription("query", 1234) subscription2 = Subscription("example", 5678) watcher = SubscriptionWatcher(api, bot) watcher._update_latest_ids(latest_submissions) watcher.subscriptions.add(subscription1) watcher.subscriptions.add(subscription2) watcher.blocklists[3452] = {"test", "example"} watcher.blocklists[1453] = {"ych"} try: watcher.save_to_json() new_watcher = SubscriptionWatcher.load_from_json(api, bot) assert len(new_watcher.latest_ids) == 3 assert "123243" in new_watcher.latest_ids assert "123242" in new_watcher.latest_ids assert "123240" in new_watcher.latest_ids assert list(watcher.latest_ids) == list(new_watcher.latest_ids) assert len(new_watcher.subscriptions) == 2 list_subs = list(new_watcher.subscriptions) if list_subs[0].query == "query": assert list_subs[0].destination == 1234 assert list_subs[1].query == "example" assert list_subs[1].destination == 5678 else: assert list_subs[0].query == "example" assert list_subs[0].destination == 5678 assert list_subs[1].query == "query" assert list_subs[1].destination == 1234 assert len(new_watcher.blocklists) == 2 assert 3452 in new_watcher.blocklists assert len(new_watcher.blocklists[3452]) == 2 assert isinstance(new_watcher.blocklists[3452], set) assert "test" in new_watcher.blocklists[3452] assert "example" in new_watcher.blocklists[3452] assert 1453 in new_watcher.blocklists assert len(new_watcher.blocklists[1453]) == 1 assert isinstance(new_watcher.blocklists[1453], set) assert "ych" in new_watcher.blocklists[1453] finally: SubscriptionWatcher.FILENAME = old_filename os.remove(test_watcher_file)
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
def test_update_latest_ids(self, bot): api = MockExportAPI() watcher = SubscriptionWatcher(api, bot) id_list = ["1234", "1233", "1230", "1229"] submissions = [MockSubmission(x) for x in id_list] mock_save_json = MockMethod() watcher.save_to_json = mock_save_json.call watcher._update_latest_ids(submissions) assert list(watcher.latest_ids) == id_list assert mock_save_json.called
def test_save_to_json(self, bot): test_watcher_file = "./test_subscription_watcher.json" if os.path.exists(test_watcher_file): os.remove(test_watcher_file) api = MockExportAPI() latest_submissions = [ SubmissionBuilder(submission_id="123243").build_short_submission(), SubmissionBuilder(submission_id="123242").build_short_submission(), SubmissionBuilder(submission_id="123240").build_short_submission() ] subscription1 = Subscription("query", 1234) subscription2 = Subscription("example", 5678) watcher = SubscriptionWatcher(api, bot) watcher._update_latest_ids(latest_submissions) watcher.subscriptions.add(subscription1) watcher.subscriptions.add(subscription2) watcher.blocklists[3452] = {"test", "example"} watcher.blocklists[1453] = {"ych"} watcher.FILENAME = test_watcher_file try: watcher.save_to_json() assert os.path.exists(test_watcher_file) with open(test_watcher_file, "r") as f: data = json.load(f) assert data is not None assert len(data['latest_ids']) == 3 assert "123240" in data['latest_ids'] assert "123242" in data['latest_ids'] assert "123243" in data['latest_ids'] assert len(data['subscriptions']) == 2 if data['subscriptions'][0]['query'] == "query": assert data['subscriptions'][0]['destination'] == 1234 assert data['subscriptions'][1]['query'] == "example" assert data['subscriptions'][1]['destination'] == 5678 else: assert data['subscriptions'][0]['query'] == "example" assert data['subscriptions'][0]['destination'] == 5678 assert data['subscriptions'][1]['query'] == "query" assert data['subscriptions'][1]['destination'] == 1234 assert len(data["blacklists"]) == 2 assert "3452" in data["blacklists"] assert len(data["blacklists"]["3452"]) == 2 assert isinstance(data["blacklists"]["3452"], list) assert "test" in data["blacklists"]["3452"] assert "example" in data["blacklists"]["3452"] assert "1453" in data["blacklists"] assert len(data["blacklists"]["1453"]) == 1 assert isinstance(data["blacklists"]["1453"], list) assert data["blacklists"]["1453"] == ["ych"] finally: os.remove(test_watcher_file)
def test_list_subs__alphabetical(context): api = MockExportAPI() watcher = SubscriptionWatcher(api, context.bot) watcher.subscriptions.add(Subscription("example", 18749)) watcher.subscriptions.add(Subscription("test", 18749)) watcher.subscriptions.add(Subscription("deer", 18749)) func = SubscriptionFunctionality(watcher) resp = func._list_subs(18749) assert "Current active subscriptions in this chat:" in resp assert "- deer\n- example\n- test" in resp