]> WPIA git - motion.git/commitdiff
Merge branch 'unit-test' into 'master'
authorFelix Dörre <felix@dogcraft.de>
Fri, 12 Jun 2020 14:37:21 +0000 (16:37 +0200)
committerFelix Dörre <felix@dogcraft.de>
Fri, 12 Jun 2020 14:37:21 +0000 (16:37 +0200)
Add Unit test

See merge request felixdoerre/motion!12

README.md
motion.py
sql/sample_data.sql [new file with mode: 0644]
tests/test_motion.py [new file with mode: 0644]

index 01406efc9962c48c8a6a706577a68ff88081ae4c..1181181cdcd7fa71104bad99afd76cc48741c264 100644 (file)
--- a/README.md
+++ b/README.md
@@ -8,11 +8,24 @@ pip install -r requirements.txt
 ```
 Then edit config.py.example into config.py with your database connection
 
-To debug-run:
+To debug-run linux:
 ```
 LANG=C.UTF-8 FLASK_DEBUG=1 FLASK_APP=motion.py flask run
 ```
 
+To debug-run windows:
+```
+set LANG=C.UTF-8
+set FLASK_DEBUG=1
+set FLASK_APP=motion.py
+flask run
+```
+
+For unit testing use config values from config.py.example:
+```
+python -m unittest tests/test_motion.py
+```
+
 The database schema is automatically installed when the table "schema_version" does not exist and the application is started.
 
 # Usage
index f6ddf1344004ee30bd5bbb40d096eadcd33cea90..df5db15b57e335e16a3cd4da5c0e6fc3edef779a 100644 (file)
--- a/motion.py
+++ b/motion.py
@@ -27,11 +27,20 @@ md = Markdown(app, extensions=[EscapeHtml()])
 # Load config
 app.config.from_pyfile('config.py')
 
-prefix=app.config.get("GROUP_PREFIX")
 
-times=app.config.get("DURATION")
+class ConfigProxy:
+    def __init__(self, name):
+        self.name = name
+    @property
+    def per_host(self):
+        dict = app.config.get(self.name)
+        if dict is None:
+            return None
+        return dict.get(request.host)
 
-debuguser=app.config.get("DEBUGUSER")
+prefix = ConfigProxy("GROUP_PREFIX")
+times = ConfigProxy("DURATION")
+debuguser = ConfigProxy("DEBUGUSER")
 
 @app.before_request
 def lookup_user():
@@ -39,8 +48,9 @@ def lookup_user():
 
     env = request.environ
     user = None
-    if debuguser is not None:
-        parts =debuguser[request.host].split("/", 1)
+    my_debuguser = debuguser.per_host
+    if my_debuguser is not None:
+        parts = my_debuguser.split("/", 1)
         user = parts[0]
         roles = parts[1]
 
@@ -78,7 +88,7 @@ def lookup_user():
             if a[0] not in g.roles:
                 g.roles[a[0]] = []
             if val == "*":
-                g.roles[a[0]] = [group for group in prefix[request.host]]
+                g.roles[a[0]] = [group for group in prefix.per_host]
             else:
                 g.roles[a[0]].append(val)
     return None
@@ -161,7 +171,7 @@ def main():
             prev = rs[9][0]
         else:
             prev = -1
-    return render_template('index.html', motions=rv[:10], more=rv[10]["id"] if len(rv) == 11 else None, times=times[request.host], prev=prev,
+    return render_template('index.html', motions=rv[:10], more=rv[10]["id"] if len(rv) == 11 else None, times=times.per_host, prev=prev,
                            categories=get_allowed_cats("create"))
 
 def rel_redirect(loc):
@@ -175,7 +185,7 @@ def put_motion():
     if cat not in get_allowed_cats("create"):
         return "Forbidden", 403
     time = int(request.form.get("days", "3"));
-    if time not in times[request.host]:
+    if time not in times.per_host:
         return "Error, invalid length", 500
     db = get_db()
     with db.xact():
@@ -184,9 +194,9 @@ def put_motion():
         sr = s(cat, request.host)
         ident=""
         if len(sr) == 0 or sr[0][0] is None:
-            ident=prefix[request.host][cat]+"."+t.strftime("%Y%m%d")+".001"
+            ident=prefix.per_host[cat]+"."+t.strftime("%Y%m%d")+".001"
         else:
-            ident=prefix[request.host][cat]+"."+t.strftime("%Y%m%d")+"."+("%03d" % (int(sr[0][0].split(".")[2])+1))
+            ident=prefix.per_host[cat]+"."+t.strftime("%Y%m%d")+"."+("%03d" % (int(sr[0][0].split(".")[2])+1))
         p = db.prepare("INSERT INTO motion(\"name\", \"content\", \"deadline\", \"posed_by\", \"type\", \"identifier\", \"host\") VALUES($1, $2, CURRENT_TIMESTAMP + $3 * interval '1 days', $4, $5, $6, $7)")
         p(request.form.get("title", ""), request.form.get("content",""), time, g.voter, cat, ident, request.host)
     return rel_redirect("/")
@@ -215,6 +225,8 @@ def show_motion(motion):
                          + "LEFT JOIN voter canceler ON canceler.id = motion.canceled_by "
                          + "WHERE motion.identifier=$1 AND motion.host=$3")
     rv = p(motion, g.voter, request.host)
+    if len(rv) == 0:
+        return "Error, Not found", 404
     votes = None
     if may("audit", rv[0].get("type")) and not rv[0].get("running") and not rv[0].get("canceled"):
         votes = get_db().prepare("SELECT vote.result, voter.email FROM vote INNER JOIN voter ON voter.id = vote.voter_id WHERE vote.motion_id=$1")(rv[0].get("id"));
diff --git a/sql/sample_data.sql b/sql/sample_data.sql
new file mode 100644 (file)
index 0000000..fd762be
--- /dev/null
@@ -0,0 +1,24 @@
+-- sample data for scheme version 3
+INSERT INTO voter (id,email) VALUES (1, 'User A');
+INSERT INTO voter (id,email) VALUES (2, 'User B');
+INSERT INTO voter (id,email) VALUES (3, 'User C');
+ALTER SEQUENCE voter_id_seq RESTART WITH 4;
+
+INSERT INTO motion (id,identifier,name,type,host,content,posed,posed_by,deadline,canceled,cancelation_reason,canceled_by) VALUES
+    (1,'g1.20200402.001','Motion A','group1','127.0.0.1:5000','My special motion','2020-04-02 21:40:33.780364',1,'2020-04-02 21:40:33.780364',Null,Null,Null);
+INSERT INTO motion (id,identifier,name,type,host,content,posed,posed_by,deadline,canceled,cancelation_reason,canceled_by) VALUES
+    (2,'g1.20200402.002','Motion B','group1','127.0.0.1:5000','A second motion','2020-04-02 21:41:26.588442',1,'2020-04-04 21:41:26.588442',Null,Null,Null);
+INSERT INTO motion (id,identifier,name,type,host,content,posed,posed_by,deadline,canceled,cancelation_reason,canceled_by) VALUES
+    (3,'g1.20200402.003','Motion C','group1','127.0.0.1:5000','A third motion', '2020-04-02 21:47:24.969588',1,'2020-04-04 21:47:24.969588','2020-04-03 21:48:24.969588','Entered with wrong text',1);
+-- add motion with timespan from now to 1 day from now
+INSERT INTO motion (id,identifier,name,type,host,content,posed,posed_by,deadline,canceled,cancelation_reason,canceled_by) VALUES
+    (4,'g1.20200402.004','Motion D','group1','127.0.0.1:5000','A fourth motion', current_timestamp ,1,current_timestamp + interval '1' day,Null,Null,Null);
+ALTER SEQUENCE motion_id_seq RESTART WITH 5;
+
+INSERT INTO vote (motion_id,voter_id,result,entered) VALUES (1,1,'yes','2020-04-02 21:54:34.469784');
+INSERT INTO vote (motion_id,voter_id,result,entered) VALUES (1,2,'yes','2020-04-02 21:54:34.469784');
+INSERT INTO vote (motion_id,voter_id,result,entered) VALUES (1,3,'no','2020-04-02 21:54:34.469784');
+INSERT INTO vote (motion_id,voter_id,result,entered) VALUES (2,1,'yes','2020-04-02 21:54:34.469784');
+INSERT INTO vote (motion_id,voter_id,result,entered) VALUES (2,2,'no','2020-04-02 21:54:34.469784');
+INSERT INTO vote (motion_id,voter_id,result,entered) VALUES (2,3,'no','2020-04-02 21:54:34.469784');
+INSERT INTO vote (motion_id,voter_id,result,entered) VALUES (3,3,'yes','2020-04-02 21:48:34.469784');
diff --git a/tests/test_motion.py b/tests/test_motion.py
new file mode 100644 (file)
index 0000000..bb4acf0
--- /dev/null
@@ -0,0 +1,457 @@
+import motion
+import unittest
+import postgresql
+from unittest import TestCase
+from motion import app
+from datetime import datetime
+
+app.config.update(
+    DEBUGUSER = {},
+    GROUP_PREFIX = {'127.0.0.1:5000': {'group1': 'g1', 'group2': 'g2'}},
+    DURATION = {'127.0.0.1:5000':[3, 7, 14]},
+    SERVER_NAME = '127.0.0.1:5000'
+)
+
+app.config['TESTING'] = True
+app.config['DEBUG'] = False
+
+
+class BasicTest(TestCase):
+
+    def init_test(self):
+        self.app = app.test_client()
+        self.assertEqual(app.debug, False)
+
+        # reset database
+        self.db_clear()
+
+    # functions to manipulate motions
+    def createVote(self, user, motion, vote):
+        return self.app.post(
+            '/motion/' + motion +'/vote',
+            environ_base={'USER_ROLES': user},
+            data=dict(vote=vote)
+        )
+        
+
+    def createMotion(self, user, motiontitle, motioncontent, days, category):
+        return self.app.post(
+            '/motion',
+            environ_base={'USER_ROLES': user},
+            data=dict(title=motiontitle, content=motioncontent, days=days, category=category)
+        )
+
+    def cancelMotion(self, user, motion, reason):
+        return self.app.post(
+            '/motion/' + motion +'/cancel',
+            environ_base={'USER_ROLES': user},
+            data=dict(reason=reason)
+        )
+
+    def buildResultText(self, motiontext, yes, no, abstain):
+        return '<p>'+motiontext+'</p></p>\n    <p>\nYes <span class=\"badge badge-pill badge-secondary\">'+str(yes)+'</span><br>'\
+            + '\nNo <span class=\"badge badge-pill badge-secondary\">'+str(no)+'</span><br>'\
+            + '\nAbstain <span class=\"badge badge-pill badge-secondary\">'+str(abstain)+'</span>'
+
+    # functions to clear database
+    def db_clear(self):
+        with postgresql.open(app.config.get("DATABASE"), user=app.config.get("USER"), password=app.config.get("PASSWORD")) as db:
+            with app.open_resource('sql/schema.sql', mode='r') as f:
+                db.execute(f.read())
+
+    def db_sampledata(self):
+        with postgresql.open(app.config.get("DATABASE"), user=app.config.get("USER"), password=app.config.get("PASSWORD")) as db:
+            with app.open_resource('sql/sample_data.sql', mode='r') as f:
+                db.execute(f.read())
+
+
+# no specific rights required
+class GeneralTests(BasicTest):
+
+    def setUp(self):
+        self.init_test()
+        global user
+        user = 'testuser/'
+        self.db_sampledata()
+
+    def tearDown(self):
+        pass
+
+    def test_main_page(self):
+        response = self.app.get('/', environ_base={'USER_ROLES': user}, follow_redirects=True)
+        self.assertEqual(response.status_code, 200)
+
+    def test_basic_results_data(self):
+        result = self.app.get('/', environ_base={'USER_ROLES': user}, follow_redirects=True)
+        testtext= '<div class="motion card" id="motion-3">\n  <div class="motion-title card-heading alert-warning">'\
+            + '\n    <span class=\"title-text\">Motion C</span> (Canceled)\n    <span class=\"motion-type\">group1</span>'\
+            + '\n    <div># <a href=\"/motion/g1.20200402.003\" class=\"anchor\">g1.20200402.003</a></div>'\
+            + '\n    <div class=\"date\">\n      <div>Proposed: 2020-04-02 21:47:24 (UTC) by User A</div>'\
+            + '\n      <div>Canceled: 2020-04-03 21:48:24 (UTC) by User A</div></div>\n  </div>'\
+            + '\n  <div class=\"card-body\">\n    <p><p>A third motion</p></p>'\
+            + '\n    <p>\nYes <span class=\"badge badge-pill badge-secondary\">1</span><br>'\
+            + '\nNo <span class=\"badge badge-pill badge-secondary\">0</span><br>'\
+            + '\nAbstain <span class=\"badge badge-pill badge-secondary\">0</span><br>\n    </p>'\
+            + '\n    <p>Cancelation reason: Entered with wrong text</p>\n  </div>\n</div>'
+        self.assertIn(str.encode(testtext), result.data)
+        testtext= '<div class="motion card" id="motion-2">\n  <div class="motion-title card-heading alert-danger">'\
+            + '\n    <span class=\"title-text\">Motion B</span> (Finished)\n    <span class=\"motion-type\">group1</span>'\
+            + '\n    <div># <a href=\"/motion/g1.20200402.002\" class=\"anchor\">g1.20200402.002</a></div>'\
+            + '\n    <div class=\"date\">\n      <div>Proposed: 2020-04-02 21:41:26 (UTC) by User A</div>'\
+            + '\n      <div>Votes until: 2020-04-04 21:41:26 (UTC)</div></div>\n  </div>'\
+            + '\n  <div class=\"card-body\">\n    <p><p>A second motion</p></p>'\
+            + '\n    <p>\nYes <span class=\"badge badge-pill badge-secondary\">1</span><br>'\
+            + '\nNo <span class=\"badge badge-pill badge-secondary\">2</span><br>'\
+            + '\nAbstain <span class=\"badge badge-pill badge-secondary\">0</span><br>\n    </p>\n  </div>\n</div>\n'
+        self.assertIn(str.encode(testtext), result.data)
+        testtext= '<div class=\"motion card\" id=\"motion-1\">\n  <div class=\"motion-title card-heading alert-success\">'\
+            + '\n    <span class=\"title-text\">Motion A</span> (Finished)\n    <span class=\"motion-type\">group1</span>'\
+            + '\n    <div># <a href=\"/motion/g1.20200402.001\" class=\"anchor\">g1.20200402.001</a></div>'\
+            + '\n    <div class=\"date">\n      <div>Proposed: 2020-04-02 21:40:33 (UTC) by User A</div>'\
+            + '\n      <div>Votes until: 2020-04-02 21:40:33 (UTC)</div></div>\n  </div>'\
+            + '\n  <div class=\"card-body\">\n    <p><p>My special motion</p></p>'\
+            + '\n    <p>\nYes <span class=\"badge badge-pill badge-secondary\">2</span><br>'\
+            + '\nNo <span class=\"badge badge-pill badge-secondary\">1</span><br>'\
+            + '\nAbstain <span class=\"badge badge-pill badge-secondary\">0</span><br>\n    </p>\n  </div>\n</div>\n</div>'
+        self.assertIn(str.encode(testtext), result.data)
+
+        # start with second motion
+        result = self.app.get('/', environ_base={'USER_ROLES': user}, query_string=dict(start=2))
+        testtext= 'id=\"motion-3\">'
+        self.assertNotIn(str.encode(testtext), result.data)
+        testtext= 'id=\"motion-2">'
+        self.assertIn(str.encode(testtext), result.data)
+        testtext= 'id=\"motion-1\">'
+        self.assertIn(str.encode(testtext), result.data)
+
+    def test_basic_results_data_details(self):
+        motion='g1.20200402.002'
+        result = self.app.get('/motion/' + motion, environ_base={'USER_ROLES': user}, follow_redirects=True)
+        testtext= '<p>A second motion</p></p>\n  </div>\n</div>\n<a href=\"/?start=2#motion-2\" class=\"btn btn-primary\">Back</a>\n</body>'
+        self.assertIn(str.encode(testtext), result.data)
+
+    def test_vote(self):
+        motion='g1.20200402.004'
+        response = self.createVote(user, motion, 'yes')
+        self.assertEqual(response.status_code, 403)
+        self.assertIn(str.encode('Forbidden'), response.data)
+
+    def test_no_user(self):
+        result = self.app.get('/', follow_redirects=True)
+        self.assertEqual(result.status_code, 500)
+        self.assertIn(str.encode('Server misconfigured'), result.data)
+
+    def test_user_invalid(self):
+        result = self.app.get('/', environ_base={'USER_ROLES': '<invalid>/'}, follow_redirects=True)
+        self.assertEqual(result.status_code, 403)
+        self.assertIn(str.encode('Access denied'), result.data)
+
+    def test_basic_env(self):
+        result = self.app.get('/', environ_base={'USER': 'testuser', 'ROLES':''}, follow_redirects=True)
+        testtext= 'id=\"motion-3\">'
+        self.assertIn(str.encode(testtext), result.data)
+
+    def test_basic_results_data_details_not_given(self):
+        motion='g1.30190402.001'
+        result = self.app.get('/motion/' + motion, environ_base={'USER_ROLES': user}, follow_redirects=True)
+        self.assertEqual(result.status_code, 404)
+        self.assertIn(str.encode('Error, Not found'), result.data)
+
+
+class VoterTests(BasicTest):
+
+    def setUp(self):
+        self.init_test()
+        global user
+        user='testuser/vote:*'
+        self.db_sampledata()
+
+    def tearDown(self):
+        pass
+
+    def test_main_page(self):
+        response = self.app.get('/', environ_base={'USER_ROLES': user}, follow_redirects=True)
+        self.assertEqual(response.status_code, 200)
+
+    def test_home_data(self):
+        result = self.app.get('/', environ_base={'USER_ROLES': user})
+        self.assertNotIn("<select class=\"float form-control\" name=\"category\">", str(result.data) )
+
+    def test_vote_yes(self):
+        motion='g1.20200402.004'
+        response = self.createVote(user, motion, 'yes')
+        self.assertEqual(response.status_code, 302)
+        result = self.app.get('/', environ_base={'USER_ROLES': user})
+        resulttext=self.buildResultText('A fourth motion', 1, 0, 0)
+        result = self.app.get('/motion/' + motion, environ_base={'USER_ROLES': user}, follow_redirects=True)
+        testtext= 'class=\"btn btn-success\" name=\"vote\" value="yes" id="vote-yes">yes</button>'
+        self.assertIn(str.encode(testtext), result.data)
+        testtext= 'class=\"btn btn-primary\" name=\"vote\" value=\"no\" id=\"vote-no\">no</button>'
+        self.assertIn(str.encode(testtext), result.data)
+        testtext= 'class=\"btn btn-primary\" name=\"vote\" value=\"abstain\" id=\"vote-abstain\">abstain</button>'
+        self.assertIn(str.encode(testtext), result.data)
+
+    def test_vote_no(self):
+        motion='g1.20200402.004'
+        response = self.createVote(user, motion, 'no')
+        self.assertEqual(response.status_code, 302)
+        result = self.app.get('/', environ_base={'USER_ROLES': user})
+        resulttext=self.buildResultText('A fourth motion', 0, 1, 0)
+        self.assertIn(str.encode(resulttext), result.data)
+        result = self.app.get('/motion/' + motion, environ_base={'USER_ROLES': user}, follow_redirects=True)
+        testtext= 'class="btn btn-primary" name="vote\" value=\"yes\" id=\"vote-yes\">yes</button>'
+        self.assertIn(str.encode(testtext), result.data)
+        testtext= 'class=\"btn btn-success\" name=\"vote\" value=\"no\" id=\"vote-no\">no</button>'
+        self.assertIn(str.encode(testtext), result.data)
+        testtext= 'class=\"btn btn-primary\" name=\"vote\" value=\"abstain\" id=\"vote-abstain\">abstain</button>'
+        self.assertIn(str.encode(testtext), result.data)
+
+    def test_vote_abstain(self):
+        motion='g1.20200402.004'
+        response = self.createVote(user, motion, 'abstain')
+        self.assertEqual(response.status_code, 302)
+        result = self.app.get('/', environ_base={'USER_ROLES': user})
+        resulttext=self.buildResultText('A fourth motion', 0, 0, 1)
+        self.assertIn(str.encode(resulttext), result.data)
+        result = self.app.get('/motion/' + motion, environ_base={'USER_ROLES': user}, follow_redirects=True)
+        testtext= 'class=\"btn btn-primary\" name=\"vote\" value=\"yes\" id=\"vote-yes\">yes</button>'
+        self.assertIn(str.encode(testtext), result.data)
+        testtext= 'class=\"btn btn-primary\" name=\"vote\" value=\"no\" id=\"vote-no\">no</button>'
+        self.assertIn(str.encode(testtext), result.data)
+        testtext= 'class=\"btn btn-success\" name=\"vote\" value=\"abstain\" id=\"vote-abstain\">abstain</button>'
+        self.assertIn(str.encode(testtext), result.data)
+
+    def test_vote_change(self):
+        motion='g1.20200402.004'
+        response = self.createVote(user, motion, 'yes')
+        self.assertEqual(response.status_code, 302)
+        result = self.app.get('/', environ_base={'USER_ROLES': user})
+        resulttext=self.buildResultText('A fourth motion', 1, 0, 0)
+        self.assertIn(str.encode(resulttext), result.data)
+        response = self.createVote(user, motion, 'no')
+        self.assertEqual(response.status_code, 302)
+        result = self.app.get('/', environ_base={'USER_ROLES': user})
+        resulttext=self.buildResultText('A fourth motion', 0, 1, 0)
+        self.assertIn(str.encode(resulttext), result.data)
+        response = self.createVote(user, motion, 'abstain')
+        self.assertEqual(response.status_code, 302)
+        result = self.app.get('/', environ_base={'USER_ROLES': user})
+        resulttext=self.buildResultText('A fourth motion', 0, 0, 1)
+        self.assertIn(str.encode(resulttext), result.data)
+
+    def test_vote_group(self):
+        motion='g1.20200402.004'
+        response = self.createVote(user, motion, 'yes')
+        self.assertEqual(response.status_code, 302)
+
+        motion='g1.20200402.004'
+        user1='testuser/vote:group1'
+        response = self.createVote(user1, motion, 'yes')
+        self.assertEqual(response.status_code, 302)
+
+        motion='g1.20200402.004'
+        user1='testuser/vote:group1 vote:group2'
+        response = self.createVote(user1, motion, 'yes')
+        self.assertEqual(response.status_code, 302)
+
+    def test_vote_wrong_group(self):
+        motion='g1.20200402.004'
+        user1='testuser/vote:group2'
+        response = self.createVote(user1, motion, 'yes')
+        self.assertEqual(response.status_code, 403)
+        self.assertIn(str.encode('Forbidden'), response.data)
+
+    def test_vote_closed(self):
+        motion='g1.20200402.002'
+        response = self.createVote(user, motion, 'abstain')
+        self.assertEqual(response.status_code, 500)
+        self.assertIn(str.encode('Error, motion deadline has passed'), response.data)
+
+    def test_vote_canceled(self):
+        motion='g1.20200402.003'
+        response = self.createVote(user, motion, 'abstain')
+        self.assertEqual(response.status_code, 500)
+        self.assertIn(str.encode('Error, motion deadline has passed'), response.data)
+
+    def test_vote_not_given(self):
+        motion='g1.30190402.001'
+        response = self.createVote(user, motion, 'abstain')
+        self.assertEqual(response.status_code, 404)
+        self.assertIn(str.encode('Error, Not found'), response.data)
+
+    def test_cancelMotion(self):
+        motion='g1.20200402.004'
+        reason="none"
+        response = self.cancelMotion(user, motion, reason)
+        self.assertEqual(response.status_code, 403)
+        self.assertIn(str.encode('Forbidden'), response.data)
+
+    def test_see_old_vote(self):
+        motion='g1.20200402.002'
+        result = self.app.get('/motion/' + motion, environ_base={'USER_ROLES': user}, follow_redirects=True)
+        testtext= '<div>Proposed: 2020-04-02 21:41:26 (UTC) by User A</div>\n      <div>Votes until: 2020-04-04 21:41:26 (UTC)</div></div>'\
+            + '\n  </div>\n  <div class="card-body">\n    <p><p>A second motion</p></p>\n  </div>\n</div>'\
+            + '\n<a href="/?start=2#motion-2" class="btn btn-primary">Back</a>'
+        self.assertIn(str.encode(testtext), result.data)
+
+    def test_createMotion(self):
+        title='My Motion'
+        content='My body'
+        response = self.createMotion(user, title, content, '3', 'group1')
+        self.assertEqual(response.status_code, 403)
+        self.assertIn(str.encode('Forbidden'), response.data)
+
+
+class CreateMotionTests(BasicTest):
+
+    def setUp(self):
+        self.init_test()
+        global user
+        user='testuser/vote:* create:* cancel:*'
+        self.db_clear()
+
+    def tearDown(self):
+        pass
+
+    def test_main_page(self):
+        response = self.app.get('/', environ_base={'USER_ROLES': user}, follow_redirects=True)
+        self.assertEqual(response.status_code, 200)
+
+    def test_home_data(self):
+        result = self.app.get('/', environ_base={'USER_ROLES': user})
+
+        # assert the response data
+        self.assertIn(b'User: testuser', result.data)
+        self.assertIn("<select class=\"float form-control\" name=\"category\">", str(result.data) )
+
+    def test_createMotion(self):
+        title='My Motion'
+        content='My body'
+        response = self.createMotion(user, title, content, '3', 'group1')
+        self.assertEqual(response.status_code, 302)
+        result = self.app.get('/', environ_base={'USER_ROLES': user})
+        self.assertIn(str.encode(title), result.data)
+        self.assertIn(str.encode(content), result.data)
+        self.assertIn(str.encode('g1.'+datetime.today().strftime('%Y%m%d')+'.001'), result.data)
+
+        title='My Motion1'
+        content='My body1'
+        response = self.createMotion(user, title, content, '3', 'group1')
+        self.assertEqual(response.status_code, 302)
+        result = self.app.get('/', environ_base={'USER_ROLES': user})
+        self.assertIn(str.encode(title), result.data)
+        self.assertIn(str.encode(content), result.data)
+        self.assertIn(str.encode('g1.'+datetime.today().strftime('%Y%m%d')+'.002'), result.data)
+
+        title='My Motion2'
+        content='My body2'
+        response = self.createMotion(user, title, content, '3', 'group2')
+        self.assertEqual(response.status_code, 302)
+        result = self.app.get('/', environ_base={'USER_ROLES': user})
+        self.assertIn(str.encode(title), result.data)
+        self.assertIn(str.encode(content), result.data)
+        self.assertIn(str.encode('g2.'+datetime.today().strftime('%Y%m%d')+'.001'), result.data)
+
+        title='My Motion3'
+        content='My body3'
+        user1='testuser/vote:* create:group1 cancel:*'
+        response = self.createMotion(user1, title, content, '3', 'group1')
+        self.assertEqual(response.status_code, 302)
+
+        title='My Motion4'
+        content='My body4'
+        user1='testuser/vote:* create:group1 create:group2 cancel:*'
+        response = self.createMotion(user1, title, content, '3', 'group1')
+        self.assertEqual(response.status_code, 302)
+
+
+    def test_createMotionMarkdown(self):
+        title='Markdown Test'
+        content= 'MyMotionBody MD [text](https//domain.tld/link)'
+        response = self.createMotion(user, title, content, '3', 'group1')
+        self.assertEqual(response.status_code, 302)
+        result = self.app.get('/', environ_base={'USER_ROLES': user})
+        self.assertIn(str.encode(title), result.data)
+        self.assertIn(b'MyMotionBody MD <a href=\"https//domain.tld/link\">text</a>', result.data)
+
+    def test_createMotionMarkdownDirectLink(self):
+        title='Markdown Test Link'
+        content='MyMotionBody MD <a href=\"https//domain.tld/link\">direct</a'
+        response = self.createMotion(user, title, content, '3', 'group1')
+        self.assertEqual(response.status_code, 302)
+        result = self.app.get('/', environ_base={'USER_ROLES': user})
+        self.assertIn(str.encode(title), result.data)
+        self.assertIn(b'MyMotionBody MD &lt;a href="https//domain.tld/link"&gt;direct&lt;/a', result.data)
+
+    def test_createMotionMarkdownCombined(self):
+        title='Markdown Test Link'
+        content='Body [combined](https//domain.tld/link) <a href=\"https//domain.tld/link\">combined1</a'
+        response = self.createMotion(user, title, content, '3', 'group1')
+        self.assertEqual(response.status_code, 302)
+        result = self.app.get('/', environ_base={'USER_ROLES': user})
+        self.assertIn(str.encode(title), result.data)
+        self.assertIn(b'Body <a href=\"https//domain.tld/link\">combined</a> &lt;a href="https//domain.tld/link"&gt;combined1&lt;/a', result.data)
+
+    def test_createMotionWrongDayLength(self):
+        title='My Motion'
+        content='My body'
+        response = self.createMotion(user, title, content, '21', 'group1')
+        self.assertEqual(response.status_code, 500)
+        self.assertIn(str.encode('Error, invalid length'), response.data)
+
+    def test_createMotionWrongGroup(self):
+        title='My Motion'
+        content='My body'
+        response = self.createMotion(user, title, content, '3', 'test1')
+        self.assertEqual(response.status_code, 403)
+        self.assertIn(str.encode('Forbidden'), response.data)
+
+        user1='testuser/vote:* create:group1 cancel:*'
+        response = self.createMotion(user1, title, content, '3', 'group2')
+        self.assertEqual(response.status_code, 403)
+        self.assertIn(str.encode('Forbidden'), response.data)
+
+    def test_cancelMotion(self):
+        self.db_sampledata()
+
+        motion='g1.20200402.004'
+        reason="none"
+        response = self.cancelMotion(user, motion, reason)
+        self.assertEqual(response.status_code, 500)
+        self.assertIn(str.encode('Error, form requires reason'), response.data)
+
+        reason='cancel test'
+        response = self.cancelMotion(user, motion, reason)
+        self.assertEqual(response.status_code, 302)
+        result = self.app.get('/', environ_base={'USER_ROLES': user})
+        self.assertIn(b'Cancelation reason: ' + str.encode(reason), result.data)
+
+        motion='g1.30190402.001'
+        reason="none"
+        response = self.cancelMotion(user, motion, reason)
+        self.assertEqual(response.status_code, 404)
+        self.assertIn(str.encode('Error, Not found'), response.data)
+
+
+class AuditMotionTests(BasicTest):
+
+    def setUp(self):
+        self.init_test()
+        global user
+        user='testuser/audit:*'
+        self.db_sampledata()
+
+    def tearDown(self):
+        pass
+
+    def test_see_old_vote(self):
+        motion='g1.20200402.002'
+        result = self.app.get('/motion/' + motion, environ_base={'USER_ROLES': user}, follow_redirects=True)
+        testtext= '<div class="motion card" id="votes">\n  <div class="card-heading text-white bg-info">\n    Motion Votes\n  </div>'\
+            + '\n  <div class="card-body">\n    <div>User A: yes</div>\n    <div>User B: no</div>'\
+            + '\n    <div>User C: no</div>\n  </div>\n</div>\n<a href="/?start=2#motion-2" class="btn btn-primary">Back</a>'
+        self.assertIn(str.encode(testtext), result.data)
+
+
+if __name__ == "__main__":
+    unittest.main()