model.py 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. import os
  2. import peewee
  3. import pytest
  4. os.environ["APP_PATH"] = "test"
  5. import lc.config as c
  6. import lc.error as e
  7. import lc.request as r
  8. import lc.model as m
  9. class Testdb:
  10. def setup_method(self, _):
  11. c.db.init(":memory:")
  12. c.db.create_tables(m.MODELS)
  13. def teardown_method(self, _):
  14. c.db.close()
  15. def mk_user(self, name="gdritter", password="foo") -> m.User:
  16. return m.User.from_request(r.User(name=name, password=password,))
  17. def test_create_user(self):
  18. name = "gdritter"
  19. u = self.mk_user(name=name)
  20. # it should be the only thing in the db
  21. all_users = m.User.select()
  22. assert len(all_users) == 1
  23. assert all_users[0].id == u.id
  24. assert all_users[0].name == name
  25. # we should be able to find it with the given name, too
  26. named_user = m.User.get(m.User.name == name)
  27. assert named_user.id == u.id
  28. assert named_user.name == name
  29. def test_user_passwords(self):
  30. name = "gdritter"
  31. password = "foo"
  32. u = self.mk_user(name=name, password=password)
  33. print(u.name, u.passhash)
  34. assert u.authenticate(password)
  35. assert u.authenticate("wrong password") is False
  36. def test_no_duplicate_users(self):
  37. name = "gdritter"
  38. u1 = self.mk_user(name=name)
  39. with pytest.raises(e.UserExists):
  40. u2 = self.mk_user(name=name)
  41. def test_get_or_create_tag(self):
  42. u = self.mk_user()
  43. tag_name = "food"
  44. t = m.Tag.get_or_create_tag(u, tag_name)
  45. # we should be able to find the tag with the given name
  46. named_tags = m.Tag.select(m.Tag.user == u and m.Tag.name == tag_name)
  47. assert len(named_tags) == 1
  48. # subsequent calls to get_or_create_tag should return the same db row
  49. t2 = m.Tag.get_or_create_tag(u, tag_name)
  50. assert t.id == t2.id
  51. def test_find_hierarchy(self):
  52. u = self.mk_user()
  53. t = m.Tag.get_or_create_tag(u, "food/bread/rye")
  54. # this should have created three db rows: for 'food', for
  55. # 'food/bread', and for 'food/bread/rye':
  56. assert len(m.Tag.select()) == 3
  57. # searching for a prefix of the tag should yield the same
  58. # parent tag
  59. assert t.parent.id == m.Tag.get(name="food/bread").id
  60. assert t.parent.parent.id == m.Tag.get(name="food").id
  61. # creating a new hierarchical tag with a shared prefix should
  62. # only create the new child tag
  63. t2 = m.Tag.get_or_create_tag(u, "food/bread/baguette")
  64. print([t.name for t in m.Tag.select()])
  65. assert len(m.Tag.select()) == 4
  66. # it should share the same parent tags
  67. assert t2.parent.id == t.parent.id
  68. assert t2.parent.parent.id == t.parent.parent.id
  69. # trying to get a hierarchical tag should result in the same
  70. # one already entered
  71. assert t.id == m.Tag.get(name="food/bread/rye").id
  72. assert t2.id == m.Tag.get(name="food/bread/baguette").id
  73. def test_create_invite(self):
  74. u = self.mk_user()
  75. invite = m.UserInvite.manufacture(u)
  76. # the invite should reference the user and be unclaimed
  77. assert invite.created_by.id == u.id
  78. assert invite.created_at is not None
  79. assert invite.claimed_by is None
  80. assert invite.claimed_at is None
  81. # deserializing the unique token should reveal the encrypted data
  82. raw_data = c.serializer.loads(invite.token)
  83. assert raw_data["created_by"] == u.name
  84. def test_use_invite(self):
  85. u = self.mk_user()
  86. initial_invite = m.UserInvite.manufacture(u)
  87. assert initial_invite.claimed_by is None
  88. assert initial_invite.claimed_at is None
  89. u2 = m.User.from_invite(r.User(name="u2", password="u2"), initial_invite.token)
  90. invite = m.UserInvite.by_code(initial_invite.token)
  91. assert invite.token == initial_invite.token
  92. assert invite.created_by.id == u.id
  93. assert invite.claimed_by.id == u2.id
  94. assert invite.created_at is not None
  95. assert invite.claimed_at is not None
  96. def bad_use_invite(self):
  97. initial_invite = m.UserInvite.manufacture(self.mk_user())
  98. # creating this user claims the invite
  99. m.User.from_invite(r.User(name="u2", password="u2"), initial_invite.token)
  100. # using the invite again raise an error
  101. with pytest.raises(e.AlreadyUsedInvite):
  102. m.User.from_invite(r.User(name="u3", password="u3"), initial_invite.token)
  103. with pytest.raises(e.NoSuchInvite):
  104. m.User.from_invite(r.User(name="u4", password="u4"), "a-non-existent-token")
  105. def check_tags(self, l, tags):
  106. present = set(map(lambda hastag: hastag.tag.name, l.tags))
  107. assert present == set(tags)
  108. def test_edit_link(self):
  109. u = self.mk_user()
  110. req = r.Link("http://foo.com", "foo", "", False, ["foo", "bar"])
  111. l = m.Link.from_request(u, req)
  112. assert l.name == req.name
  113. assert l.tags == ["foo", "bar"] # type: ignore
  114. # check the in-place update
  115. req.name = "bar"
  116. req.tags = ["bar", "baz"]
  117. req.private = True
  118. l.update_from_request(u, req)
  119. assert l.name == req.name
  120. assert l.private
  121. assert l.created != req.created
  122. self.check_tags(l, req.tags)
  123. # check that the link was persisted
  124. l2 = m.Link.by_id(l.id)
  125. assert l2
  126. assert l2.name == req.name
  127. assert l2.private
  128. assert l2.created != req.created
  129. self.check_tags(l2, req.tags)