diff --git a/OpenOversight/app/main/views.py b/OpenOversight/app/main/views.py index 53a403ac3..211c75278 100644 --- a/OpenOversight/app/main/views.py +++ b/OpenOversight/app/main/views.py @@ -366,10 +366,7 @@ def sitemap_officers(): yield "main.officer_profile", {"officer_id": officer.id} -@main.route( - "/officer//assignment/new", - methods=[HTTPMethod.GET, HTTPMethod.POST], -) +@main.route("/officer//assignment/new", methods=[HTTPMethod.POST]) @ac_or_admin_required def redirect_add_assignment(officer_id: int): return redirect( @@ -1059,7 +1056,7 @@ def list_officer( @main.route("/department//ranks") -def redirect_get_dept_ranks(department_id: int = 0, is_sworn_officer: bool = False): +def redirect_get_dept_ranks(department_id: int, is_sworn_officer: bool = False): flash(FLASH_MSG_PERMANENT_REDIRECT) return redirect( url_for( @@ -1073,7 +1070,7 @@ def redirect_get_dept_ranks(department_id: int = 0, is_sworn_officer: bool = Fal @main.route("/departments//ranks") @main.route("/ranks") -def get_dept_ranks(department_id: int = 0, is_sworn_officer: bool = False): +def get_dept_ranks(department_id: Optional[int] = None, is_sworn_officer: bool = False): if not department_id: department_id = request.args.get("department_id") if request.args.get("is_sworn_officer"): @@ -1098,17 +1095,17 @@ def get_dept_ranks(department_id: int = 0, is_sworn_officer: bool = False): @main.route("/department//units") -def redirect_get_dept_units(department_id: int = 0): +def redirect_get_dept_units(department_id: int): flash(FLASH_MSG_PERMANENT_REDIRECT) return redirect( - url_for("main.get_dept_ranks", department_id=department_id), + url_for("main.get_dept_units", department_id=department_id), code=HTTPStatus.PERMANENT_REDIRECT, ) @main.route("/departments//units") @main.route("/units") -def get_dept_units(department_id: int = 0): +def get_dept_units(department_id: Optional[int] = None): if not department_id: department_id = request.args.get("department_id") @@ -1292,7 +1289,6 @@ def delete_tag(tag_id: int): @main.route("/tag/set_featured/", methods=[HTTPMethod.POST]) @login_required -@ac_or_admin_required def redirect_set_featured_tag(tag_id: int): flash(FLASH_MSG_PERMANENT_REDIRECT) return redirect( @@ -1342,7 +1338,7 @@ def leaderboard(): @main.route( - "/cop_face/department//images/", + "/cop_face/department//image/", methods=[HTTPMethod.GET, HTTPMethod.POST], ) @main.route("/cop_face/image/", methods=[HTTPMethod.GET, HTTPMethod.POST]) @@ -1352,7 +1348,9 @@ def leaderboard(): ) @main.route("/cop_face/", methods=[HTTPMethod.GET, HTTPMethod.POST]) @login_required -def redirect_label_data(department_id: int = 0, image_id: int = 0): +def redirect_label_data( + department_id: Optional[int] = None, image_id: Optional[int] = None +): flash(FLASH_MSG_PERMANENT_REDIRECT) return redirect( url_for("main.label_data", department_id=department_id, image_id=image_id), @@ -1373,7 +1371,7 @@ def redirect_label_data(department_id: int = 0, image_id: int = 0): ) @main.route("/cop_faces/", methods=[HTTPMethod.GET, HTTPMethod.POST]) @login_required -def label_data(department_id: int = 0, image_id: int = 0): +def label_data(department_id: Optional[int] = None, image_id: Optional[int] = None): if department_id: department = Department.query.filter_by(id=department_id).one() if image_id: @@ -1842,8 +1840,7 @@ def submit_officer_images(officer_id: int): "/upload/department//officer/", methods=[HTTPMethod.POST], ) -@login_required -def redirect_upload(department_id: int = 0, officer_id: int = 0): +def redirect_upload(department_id: int, officer_id: Optional[int] = None): return redirect( url_for("main.upload", department_id=department_id, officer_id=officer_id), code=HTTPStatus.PERMANENT_REDIRECT, @@ -1855,9 +1852,8 @@ def redirect_upload(department_id: int = 0, officer_id: int = 0): "/upload/departments//officers/", methods=[HTTPMethod.POST], ) -@login_required @limiter.limit("250/minute") -def upload(department_id: int = 0, officer_id: int = 0): +def upload(department_id: int, officer_id: Optional[int] = None): if officer_id: officer = Officer.query.filter_by(id=officer_id).first() if not officer: diff --git a/OpenOversight/app/models/users.py b/OpenOversight/app/models/users.py index a8f52319a..ce6522913 100644 --- a/OpenOversight/app/models/users.py +++ b/OpenOversight/app/models/users.py @@ -4,5 +4,7 @@ class AnonymousUser(AnonymousUserMixin): + id = None + def is_admin_or_coordinator(self, department: Department) -> bool: return False diff --git a/OpenOversight/app/templates/description_detail.html b/OpenOversight/app/templates/description_detail.html index 6ae185488..122e0ce30 100644 --- a/OpenOversight/app/templates/description_detail.html +++ b/OpenOversight/app/templates/description_detail.html @@ -1,12 +1,11 @@ {% extends "base.html" %} -{% block page_title %} - Description Details -{% endblock page_title %} -{% block form %} -

- For officer with OOID {{ form.officer_id.data }}. -
- {{ form.description }} -

- {{ super() }} -{% endblock form %} +{% block content %} +
+ +

+

{{ obj.text_contents | markdown }}

+

+
+{% endblock content %} diff --git a/OpenOversight/app/templates/note_delete.html b/OpenOversight/app/templates/note_delete.html index 0d858a1e5..c424979a5 100644 --- a/OpenOversight/app/templates/note_delete.html +++ b/OpenOversight/app/templates/note_delete.html @@ -8,7 +8,7 @@

Delete Note on officer {{ obj.officer_id }}

Are you sure you want to delete this note? This cannot be undone. -

diff --git a/OpenOversight/app/templates/note_detail.html b/OpenOversight/app/templates/note_detail.html index 5c6e817de..a2e2dd527 100644 --- a/OpenOversight/app/templates/note_detail.html +++ b/OpenOversight/app/templates/note_detail.html @@ -1,12 +1,11 @@ {% extends "base.html" %} -{% block page_title %} - Note Details -{% endblock page_title %} -{% block form %} -

- For officer with OOID {{ form.officer_id.data }}. -
- {{ form.description }} -

- {{ super() }} -{% endblock form %} +{% block content %} +
+ +

+

{{ obj.text_contents | markdown }}

+

+
+{% endblock content %} diff --git a/OpenOversight/tests/test_functional.py b/OpenOversight/tests/test_functional.py index befefb10e..25e7ff741 100644 --- a/OpenOversight/tests/test_functional.py +++ b/OpenOversight/tests/test_functional.py @@ -40,6 +40,19 @@ def login_admin(browser, server_port): wait_for_element(browser, By.TAG_NAME, "body") +def logout(browser, server_port): + browser.get(f"http://localhost:{server_port}/auth/logout") + wait_for_page_load(browser) + + +def submit_image_to_dropzone(browser, img_path): + # Submit files in selenium: https://stackoverflow.com/a/61566075 + wait_for_element(browser, By.CLASS_NAME, "dz-hidden-input") + upload = browser.find_element(By.CLASS_NAME, "dz-hidden-input") + upload.send_keys(img_path) + wait_for_element(browser, By.CLASS_NAME, "dz-success") + + def wait_for_element(browser, locator, text, timeout=10): try: element_present = expected_conditions.presence_of_element_located( @@ -359,12 +372,7 @@ def test_image_classification_and_tagging(mockdata, browser, server_port): wait_for_page_load(browser) Select(browser.find_element("id", "department")).select_by_value(dept_id) - - # Submit files in selenium: https://stackoverflow.com/a/61566075 - wait_for_element(browser, By.CLASS_NAME, "dz-hidden-input") - upload = browser.find_element(By.CLASS_NAME, "dz-hidden-input") - upload.send_keys(img_path) - wait_for_element(browser, By.CLASS_NAME, "dz-success") + submit_image_to_dropzone(browser, img_path) # 4. Classify the uploaded image browser.get(f"http://localhost:{server_port}/sort/departments/{dept_id}") @@ -392,8 +400,7 @@ def test_image_classification_and_tagging(mockdata, browser, server_port): assert "Tag added to database" in page_text # 6. Log out as admin - browser.get(f"http://localhost:{server_port}/auth/logout") - wait_for_page_load(browser) + logout(browser, server_port) # 7. Check that the tag appears on the officer page browser.get(f"http://localhost:{server_port}/officers/{officer_id}") @@ -417,3 +424,50 @@ def test_image_classification_and_tagging(mockdata, browser, server_port): >= frame.location["y"] + frame.size["height"] ) assert image.location["y"] <= frame.location["y"] + + +@pytest.mark.skip("Enable once real file upload in tests is supported.") +def test_anonymous_user_can_upload_image(mockdata, browser, server_port): + test_dir = os.path.dirname(os.path.realpath(__file__)) + img_path = os.path.join(test_dir, "images/200Cat.jpeg") + + login_admin(browser, server_port) + + # 1. Create new department as admin (to avoid mockdata) + browser.get(f"http://localhost:{server_port}/departments/new") + wait_for_page_load(browser) + browser.find_element(By.ID, "name").send_keys("Auburn Police Department") + browser.find_element(By.ID, "short_name").send_keys("APD") + Select(browser.find_element(By.ID, "state")).select_by_value("WA") + browser.find_element(By.ID, "submit").click() + wait_for_page_load(browser) + + # 2. Log out + logout(browser, server_port) + + # 3. Upload image + browser.get(f"http://localhost:{server_port}/submit") + wait_for_page_load(browser) + + dept_select = Select(browser.find_element("id", "department")) + dept_select.select_by_visible_text("[WA] Auburn Police Department") + dept_id = dept_select.first_selected_option.get_attribute("value") + + submit_image_to_dropzone(browser, img_path) + + # 4. Login as admin again + login_admin(browser, server_port) + + # 5. Check that there is 1 image to classify + browser.get(f"http://localhost:{server_port}/sort/departments/{dept_id}") + wait_for_page_load(browser) + + page_text = browser.find_element(By.TAG_NAME, "body").text + assert "Do you see uniformed law enforcement officers in the photo?" in page_text + + browser.find_element(By.ID, "answer-yes").click() + wait_for_page_load(browser) + + # 6. All images tagged! + page_text = browser.find_element(By.TAG_NAME, "body").text + assert "All images have been classified!" in page_text