from django.conf import settings
from django.test import Client, override_settings
from django.urls import reverse
from django.utils.text import slugify
from evennia.help import filehelp
from evennia.utils import class_from_module
from evennia.utils.create import create_help_entry
from evennia.utils.test_resources import BaseEvenniaTest
_FILE_HELP_ENTRIES = None
[docs]class EvenniaWebTest(BaseEvenniaTest):
# Use the same classes the views are expecting
account_typeclass = settings.BASE_ACCOUNT_TYPECLASS
object_typeclass = settings.BASE_OBJECT_TYPECLASS
character_typeclass = settings.BASE_CHARACTER_TYPECLASS
exit_typeclass = settings.BASE_EXIT_TYPECLASS
room_typeclass = settings.BASE_ROOM_TYPECLASS
script_typeclass = settings.BASE_SCRIPT_TYPECLASS
channel_typeclass = settings.BASE_CHANNEL_TYPECLASS
# Default named url
url_name = "index"
# Response to expect for unauthenticated requests
unauthenticated_response = 200
# Response to expect for authenticated requests
authenticated_response = 200
[docs] def setUp(self):
super().setUp()
# Add chars to account rosters
self.account.characters.add(self.char1)
self.account2.characters.add(self.char2)
for account in (self.account, self.account2):
# Demote accounts to Player permissions
account.permissions.add("Player")
account.permissions.remove("Developer")
# Grant permissions to chars
for char in account.characters:
char.locks.add("edit:id(%s) or perm(Admin)" % account.pk)
char.locks.add("delete:id(%s) or perm(Admin)" % account.pk)
char.locks.add("view:all()")
[docs] def test_valid_chars(self):
"Make sure account has playable characters"
self.assertTrue(self.char1 in self.account.characters)
self.assertTrue(self.char2 in self.account2.characters)
[docs] def get_kwargs(self):
return {}
[docs] def test_get(self):
# Try accessing page while not logged in
response = self.client.get(reverse(self.url_name, kwargs=self.get_kwargs()))
self.assertEqual(response.status_code, self.unauthenticated_response)
[docs] def login(self):
return self.client.login(username="TestAccount", password="testpassword")
[docs] def test_get_authenticated(self):
logged_in = self.login()
self.assertTrue(logged_in, "Account failed to log in!")
# Try accessing page while logged in
response = self.client.get(reverse(self.url_name, kwargs=self.get_kwargs()), follow=True)
self.assertEqual(response.status_code, self.authenticated_response)
# ------------------------------------------------------------------------------
[docs]class AdminTest(EvenniaWebTest):
url_name = "django_admin"
unauthenticated_response = 302
[docs]class IndexTest(EvenniaWebTest):
url_name = "index"
[docs]class RegisterTest(EvenniaWebTest):
url_name = "register"
[docs]class LoginTest(EvenniaWebTest):
url_name = "login"
[docs]class LogoutTest(EvenniaWebTest):
url_name = "logout"
[docs]class PasswordResetTest(EvenniaWebTest):
url_name = "password_change"
unauthenticated_response = 302
[docs]class WebclientTest(EvenniaWebTest):
url_name = "webclient:index"
[docs] @override_settings(WEBCLIENT_ENABLED=True)
def test_get(self):
self.authenticated_response = 200
self.unauthenticated_response = 200
super().test_get()
[docs] @override_settings(WEBCLIENT_ENABLED=False)
def test_get_disabled(self):
self.authenticated_response = 404
self.unauthenticated_response = 404
super().test_get()
[docs]class ChannelListTest(EvenniaWebTest):
url_name = "channels"
[docs]class ChannelDetailTest(EvenniaWebTest):
url_name = "channel-detail"
[docs] def setUp(self):
super().setUp()
klass = class_from_module(
self.channel_typeclass, fallback=settings.FALLBACK_CHANNEL_TYPECLASS
)
# Create a channel
klass.create("demo")
[docs] def get_kwargs(self):
return {"slug": slugify("demo")}
[docs]class HelpListTest(EvenniaWebTest):
url_name = "help"
HELP_ENTRY_DICTS = [
{"key": "unit test file entry", "category": "General", "text": "cache test file entry text"}
]
[docs]class HelpDetailTest(EvenniaWebTest):
url_name = "help-entry-detail"
[docs] def setUp(self):
super().setUp()
# create a db help entry
create_help_entry("unit test db entry", "unit test db entry text", category="General")
[docs] def get_kwargs(self):
return {"category": slugify("general"), "topic": slugify("unit test db entry")}
[docs] def test_view(self):
response = self.client.get(reverse(self.url_name, kwargs=self.get_kwargs()), follow=True)
self.assertEqual(response.context["entry_text"], "unit test db entry text")
[docs] def test_object_cache(self):
# clear file help entries, use local HELP_ENTRY_DICTS to recreate new entries
global _FILE_HELP_ENTRIES
if _FILE_HELP_ENTRIES is None:
from evennia.help.filehelp import FILE_HELP_ENTRIES as _FILE_HELP_ENTRIES
help_module = "evennia.web.website.tests"
self.file_help_store = _FILE_HELP_ENTRIES.__init__(help_file_modules=[help_module])
# request access to an entry
response = self.client.get(reverse(self.url_name, kwargs=self.get_kwargs()), follow=True)
self.assertEqual(response.context["entry_text"], "unit test db entry text")
# request a second entry, verifing the cached object is not provided on a new topic request
entry_two_args = {"category": slugify("general"), "topic": slugify("unit test file entry")}
response = self.client.get(reverse(self.url_name, kwargs=entry_two_args), follow=True)
self.assertEqual(response.context["entry_text"], "cache test file entry text")
[docs]class HelpLockedDetailTest(EvenniaWebTest):
url_name = "help-entry-detail"
[docs] def setUp(self):
super().setUp()
# create a db entry with a lock
self.db_help_entry = create_help_entry(
"unit test locked topic",
"unit test locked entrytext",
category="General",
locks="read:perm(Developer)",
)
[docs] def get_kwargs(self):
return {"category": slugify("general"), "topic": slugify("unit test locked topic")}
[docs] def test_locked_entry(self):
# request access to an entry for permission the account does not have
response = self.client.get(reverse(self.url_name, kwargs=self.get_kwargs()), follow=True)
self.assertEqual(response.context["entry_text"], "Failed to find entry.")
[docs] def test_lock_with_perm(self):
# log TestAccount in, grant permission required, read the entry
self.login()
self.account.permissions.add("Developer")
response = self.client.get(reverse(self.url_name, kwargs=self.get_kwargs()), follow=True)
self.assertEqual(response.context["entry_text"], "unit test locked entrytext")
[docs]class CharacterCreateView(EvenniaWebTest):
url_name = "character-create"
unauthenticated_response = 302
[docs] @override_settings(MAX_NR_CHARACTERS=1)
def test_valid_access_multisession_0(self):
"Account1 with no characters should be able to create a new one"
self.account.characters.remove(self.char1)
# Login account
self.login()
# Post data for a new character
data = {"db_key": "gannon", "desc": "Some dude."}
response = self.client.post(reverse(self.url_name), data=data, follow=True)
self.assertEqual(response.status_code, 200)
# Make sure the character was actually created
self.assertTrue(
len(self.account.characters) == 1,
"Account only has the following characters attributed to it: %s"
% self.account.characters,
)
[docs] @override_settings(MAX_NR_CHARACTERS=5)
def test_valid_access_multisession_2(self):
"Account1 should be able to create multiple new characters"
# Login account
self.login()
# Post data for a new character
data = {"db_key": "gannon", "desc": "Some dude."}
response = self.client.post(reverse(self.url_name), data=data, follow=True)
self.assertEqual(response.status_code, 200)
# Make sure the character was actually created
self.assertTrue(
len(self.account.characters) > 1,
"Account only has the following characters attributed to it: %s"
% self.account.characters,
)
[docs]class CharacterPuppetView(EvenniaWebTest):
url_name = "character-puppet"
unauthenticated_response = 302
[docs] def get_kwargs(self):
return {"pk": self.char1.pk, "slug": slugify(self.char1.name)}
[docs] def test_invalid_access(self):
"Account1 should not be able to puppet Account2:Char2"
# Login account
self.login()
# Try to access puppet page for char2
kwargs = {"pk": self.char2.pk, "slug": slugify(self.char2.name)}
response = self.client.get(reverse(self.url_name, kwargs=kwargs), follow=True)
self.assertTrue(
response.status_code >= 400,
"Invalid access should return a 4xx code-- either obj not found or permission denied!"
" (Returned %s)" % response.status_code,
)
[docs]class CharacterListView(EvenniaWebTest):
url_name = "characters"
unauthenticated_response = 302
[docs]class CharacterManageView(EvenniaWebTest):
url_name = "character-manage"
unauthenticated_response = 302
[docs]class CharacterUpdateView(EvenniaWebTest):
url_name = "character-update"
unauthenticated_response = 302
[docs] def get_kwargs(self):
return {"pk": self.char1.pk, "slug": slugify(self.char1.name)}
[docs] def test_valid_access(self):
"Account1 should be able to update Account1:Char1"
# Login account
self.login()
# Try to access update page for char1
response = self.client.get(reverse(self.url_name, kwargs=self.get_kwargs()), follow=True)
self.assertEqual(response.status_code, 200)
# Try to update char1 desc
data = {"db_key": self.char1.db_key, "desc": "Just a regular type of dude."}
response = self.client.post(
reverse(self.url_name, kwargs=self.get_kwargs()), data=data, follow=True
)
self.assertEqual(response.status_code, 200)
# Make sure the change was made successfully
self.assertEqual(self.char1.db.desc, data["desc"])
[docs] def test_invalid_access(self):
"Account1 should not be able to update Account2:Char2"
# Login account
self.login()
# Try to access update page for char2
kwargs = {"pk": self.char2.pk, "slug": slugify(self.char2.name)}
response = self.client.get(reverse(self.url_name, kwargs=kwargs), follow=True)
self.assertEqual(response.status_code, 403)
[docs]class CharacterDeleteView(EvenniaWebTest):
url_name = "character-delete"
unauthenticated_response = 302
[docs] def get_kwargs(self):
return {"pk": self.char1.pk, "slug": slugify(self.char1.name)}
[docs] def test_valid_access(self):
"Account1 should be able to delete Account1:Char1"
# Login account
self.login()
# Try to access delete page for char1
response = self.client.get(reverse(self.url_name, kwargs=self.get_kwargs()), follow=True)
self.assertEqual(response.status_code, 200)
# Proceed with deleting it
data = {"value": "yes"}
response = self.client.post(
reverse(self.url_name, kwargs=self.get_kwargs()), data=data, follow=True
)
self.assertEqual(response.status_code, 200)
# Make sure it deleted
self.assertFalse(
self.char1 in self.account.characters,
"Char1 is still in Account playable characters list.",
)
[docs] def test_invalid_access(self):
"Account1 should not be able to delete Account2:Char2"
# Login account
self.login()
# Try to access delete page for char2
kwargs = {"pk": self.char2.pk, "slug": slugify(self.char2.name)}
response = self.client.get(reverse(self.url_name, kwargs=kwargs), follow=True)
self.assertEqual(response.status_code, 403)