# -*- coding: utf-8 -*- import os import shutil import tempfile import unittest import logging from pprint import pprint import mock from mock import MagicMock import AnkiServer from AnkiServer.collection import CollectionManager from AnkiServer.apps.rest_app import RestApp, RestHandlerRequest, CollectionHandler, ImportExportHandler, NoteHandler, ModelHandler, DeckHandler, CardHandler from webob.exc import * import anki import anki.storage class RestAppTest(unittest.TestCase): def setUp(self): self.temp_dir = tempfile.mkdtemp() self.collection_manager = CollectionManager() self.rest_app = RestApp(self.temp_dir, collection_manager=self.collection_manager) # disable all but critical errors! logging.disable(logging.CRITICAL) def tearDown(self): self.collection_manager.shutdown() self.collection_manager = None self.rest_app = None shutil.rmtree(self.temp_dir) def test_parsePath(self): tests = [ ('collection/user', ('collection', 'index', ['user'])), ('collection/user/handler', ('collection', 'handler', ['user'])), ('collection/user/note/123', ('note', 'index', ['user', '123'])), ('collection/user/note/123/handler', ('note', 'handler', ['user', '123'])), ('collection/user/deck/name', ('deck', 'index', ['user', 'name'])), ('collection/user/deck/name/handler', ('deck', 'handler', ['user', 'name'])), #('collection/user/deck/name/card/123', ('card', 'index', ['user', 'name', '123'])), #('collection/user/deck/name/card/123/handler', ('card', 'handler', ['user', 'name', '123'])), ('collection/user/card/123', ('card', 'index', ['user', '123'])), ('collection/user/card/123/handler', ('card', 'handler', ['user', '123'])), # the leading slash should make no difference! ('/collection/user', ('collection', 'index', ['user'])), ] for path, result in tests: self.assertEqual(self.rest_app._parsePath(path), result) def test_parsePath_not_found(self): tests = [ 'bad', 'bad/oaeu', 'collection', 'collection/user/handler/bad', '', '/', ] for path in tests: self.assertRaises(HTTPNotFound, self.rest_app._parsePath, path) def test_getCollectionPath(self): def fullpath(collection_id): return os.path.normpath(os.path.join(self.temp_dir, collection_id, 'collection.anki2')) # This is simple and straight forward! self.assertEqual(self.rest_app._getCollectionPath('user'), fullpath('user')) # These are dangerous - the user is trying to hack us! dangerous = ['../user', '/etc/passwd', '/tmp/aBaBaB', '/root/.ssh/id_rsa'] for collection_id in dangerous: self.assertRaises(HTTPBadRequest, self.rest_app._getCollectionPath, collection_id) def test_getHandler(self): def handlerOne(): pass def handlerTwo(): pass handlerTwo.hasReturnValue = False self.rest_app.add_handler('collection', 'handlerOne', handlerOne) self.rest_app.add_handler('deck', 'handlerTwo', handlerTwo) (handler, hasReturnValue) = self.rest_app._getHandler('collection', 'handlerOne') self.assertEqual(handler, handlerOne) self.assertEqual(hasReturnValue, True) (handler, hasReturnValue) = self.rest_app._getHandler('deck', 'handlerTwo') self.assertEqual(handler, handlerTwo) self.assertEqual(hasReturnValue, False) # try some bad handler names and types self.assertRaises(HTTPNotFound, self.rest_app._getHandler, 'collection', 'nonExistantHandler') self.assertRaises(HTTPNotFound, self.rest_app._getHandler, 'nonExistantType', 'handlerOne') def test_parseRequestBody(self): req = MagicMock() req.body = '{"key":"value"}' data = self.rest_app._parseRequestBody(req) self.assertEqual(data, {'key': 'value'}) self.assertEqual(data.keys(), ['key']) self.assertEqual(type(data.keys()[0]), str) # test some bad data req.body = '{aaaaaaa}' self.assertRaises(HTTPBadRequest, self.rest_app._parseRequestBody, req) class CollectionTestBase(unittest.TestCase): """Parent class for tests that need a collection set up and torn down.""" def setUp(self): self.temp_dir = tempfile.mkdtemp() self.collection_path = os.path.join(self.temp_dir, 'collection.anki2'); self.collection = anki.storage.Collection(self.collection_path) self.mock_app = MagicMock() def tearDown(self): self.collection.close() self.collection = None shutil.rmtree(self.temp_dir) self.mock_app.reset_mock() def add_note(self, data): from anki.notes import Note model = self.collection.models.byName(data['model']) note = Note(self.collection, model) for name, value in data['fields'].items(): note[name] = value if data.has_key('tags'): note.setTagsFromStr(data['tags']) self.collection.addNote(note) def add_default_note(self, count=1): data = { 'model': 'Basic', 'fields': { 'Front': 'The front', 'Back': 'The back', }, 'tags': "Tag1 Tag2", } for idx in range(0, count): self.add_note(data) class CollectionHandlerTest(CollectionTestBase): def setUp(self): super(CollectionHandlerTest, self).setUp() self.handler = CollectionHandler() def execute(self, name, data): ids = ['collection_name'] func = getattr(self.handler, name) req = RestHandlerRequest(self.mock_app, data, ids, {}) return func(self.collection, req) def test_list_decks(self): data = {} ret = self.execute('list_decks', data) # It contains only the 'Default' deck self.assertEqual(len(ret), 1) self.assertEqual(ret[0]['name'], 'Default') def test_select_deck(self): data = {'deck_id': '1'} ret = self.execute('select_deck', data) self.assertEqual(ret, None); def test_create_dynamic_deck_simple(self): self.add_default_note(5) data = { 'name': 'Dyn deck', 'mode': 'random', 'count': 2, 'query': "deck:\"Default\" (tag:'Tag1' or tag:'Tag2') (-tag:'Tag3')", } ret = self.execute('create_dynamic_deck', data) self.assertEqual(ret['name'], 'Dyn deck') self.assertEqual(ret['dyn'], True) cards = self.collection.findCards('deck:"Dyn deck"') self.assertEqual(len(cards), 2) def test_list_models(self): data = {} ret = self.execute('list_models', data) # get a sorted name list that we can actually check names = [model['name'] for model in ret] names.sort() # These are the default models created by Anki in a new collection default_models = [ 'Basic', 'Basic (and reversed card)', 'Basic (optional reversed card)', 'Cloze' ] self.assertEqual(names, default_models) def test_find_model_by_name(self): data = {'model': 'Basic'} ret = self.execute('find_model_by_name', data) self.assertEqual(ret['name'], 'Basic') def test_find_notes(self): ret = self.execute('find_notes', {}) self.assertEqual(ret, []) # add a note programatically self.add_default_note() # get the id for the one note on this collection note_id = self.collection.findNotes('')[0] ret = self.execute('find_notes', {}) self.assertEqual(ret, [{'id': note_id}]) ret = self.execute('find_notes', {'query': 'tag:Tag1'}) self.assertEqual(ret, [{'id': note_id}]) ret = self.execute('find_notes', {'query': 'tag:TagX'}) self.assertEqual(ret, []) ret = self.execute('find_notes', {'preload': True}) self.assertEqual(len(ret), 1) self.assertEqual(ret[0]['id'], note_id) self.assertEqual(ret[0]['model']['name'], 'Basic') def test_add_note(self): # make sure there are no notes (yet) self.assertEqual(self.collection.findNotes(''), []) # add a note programatically note = { 'model': 'Basic', 'fields': { 'Front': 'The front', 'Back': 'The back', }, 'tags': "Tag1 Tag2", } self.execute('add_note', note) notes = self.collection.findNotes('') self.assertEqual(len(notes), 1) note_id = notes[0] note = self.collection.getNote(note_id) self.assertEqual(note.model()['name'], 'Basic') self.assertEqual(note['Front'], 'The front') self.assertEqual(note['Back'], 'The back') self.assertEqual(note.tags, ['Tag1', 'Tag2']) def test_list_tags(self): ret = self.execute('list_tags', {}) self.assertEqual(ret, []) self.add_default_note() ret = self.execute('list_tags', {}) ret.sort() self.assertEqual(ret, ['Tag1', 'Tag2']) def test_set_language(self): import anki.lang self.assertEqual(anki.lang._('Again'), 'Again') try: data = {'code': 'pl'} self.execute('set_language', data) self.assertEqual(anki.lang._('Again'), u'Znowu') finally: # return everything to normal! anki.lang.setLang('en') def test_reset_scheduler(self): self.add_default_note(3) ret = self.execute('reset_scheduler', {'deck': 'Default'}) self.assertEqual(ret, { 'new_cards': 3, 'learning_cards': 0, 'review_cards': 0, }) def test_next_card(self): ret = self.execute('next_card', {}) self.assertEqual(ret, None) # add a note programatically self.add_default_note() # get the id for the one card and note on this collection note_id = self.collection.findNotes('')[0] card_id = self.collection.findCards('')[0] self.collection.sched.reset() ret = self.execute('next_card', {}) self.assertEqual(ret['id'], card_id) self.assertEqual(ret['nid'], note_id) self.assertEqual(ret['question'], 'The front') self.assertEqual(ret['answer'], 'The front\n\n