mirror of
https://gitlab.archlinux.org/archlinux/aurweb.git
synced 2025-02-03 10:43:03 +01:00
In terms of performance, most queries on this page win over PHP in query times, with the exception of sorting by Voted or Notify (https://gitlab.archlinux.org/archlinux/aurweb/-/issues/102). Otherwise, there are a few modifications: described below. * Pagination * The `paginate` Python module has been used in the FastAPI project here to implement paging on the packages search page. This changes how pagination is displayed, however it serves the same purpose. We'll take advantage of this module in other places as well. * Form action * The form action for actions now use `POST /packages` to perform. This is currently implemented and will be addressed in a follow-up commit. * Input names and values * Input names and values have been modified to satisfy the snake_case naming convention we'd like to use as much as possible. * Some input names and values were modified to comply with FastAPI Forms: (IDs[<id>]) -> (IDs, <id>). Signed-off-by: Kevin Morris <kevr@0cost.org>
914 lines
32 KiB
Python
914 lines
32 KiB
Python
import re
|
|
|
|
from datetime import datetime
|
|
from http import HTTPStatus
|
|
from typing import List
|
|
|
|
import pytest
|
|
|
|
from fastapi.testclient import TestClient
|
|
|
|
from aurweb import asgi, db
|
|
from aurweb.models.account_type import USER_ID, AccountType
|
|
from aurweb.models.dependency_type import DependencyType
|
|
from aurweb.models.official_provider import OfficialProvider
|
|
from aurweb.models.package import Package
|
|
from aurweb.models.package_base import PackageBase
|
|
from aurweb.models.package_comaintainer import PackageComaintainer
|
|
from aurweb.models.package_comment import PackageComment
|
|
from aurweb.models.package_dependency import PackageDependency
|
|
from aurweb.models.package_keyword import PackageKeyword
|
|
from aurweb.models.package_notification import PackageNotification
|
|
from aurweb.models.package_relation import PackageRelation
|
|
from aurweb.models.package_request import PackageRequest
|
|
from aurweb.models.package_vote import PackageVote
|
|
from aurweb.models.relation_type import PROVIDES_ID, RelationType
|
|
from aurweb.models.request_type import DELETION_ID, RequestType
|
|
from aurweb.models.user import User
|
|
from aurweb.testing import setup_test_db
|
|
from aurweb.testing.html import parse_root
|
|
from aurweb.testing.requests import Request
|
|
|
|
|
|
def package_endpoint(package: Package) -> str:
|
|
return f"/packages/{package.Name}"
|
|
|
|
|
|
def create_package(pkgname: str, maintainer: User) -> Package:
|
|
pkgbase = db.create(PackageBase,
|
|
Name=pkgname,
|
|
Maintainer=maintainer)
|
|
return db.create(Package, Name=pkgbase.Name, PackageBase=pkgbase)
|
|
|
|
|
|
def create_package_dep(package: Package, depname: str,
|
|
dep_type_name: str = "depends") -> PackageDependency:
|
|
dep_type = db.query(DependencyType,
|
|
DependencyType.Name == dep_type_name).first()
|
|
return db.create(PackageDependency,
|
|
DependencyType=dep_type,
|
|
Package=package,
|
|
DepName=depname)
|
|
|
|
|
|
def create_package_rel(package: Package,
|
|
relname: str) -> PackageRelation:
|
|
rel_type = db.query(RelationType,
|
|
RelationType.ID == PROVIDES_ID).first()
|
|
return db.create(PackageRelation,
|
|
RelationType=rel_type,
|
|
Package=package,
|
|
RelName=relname)
|
|
|
|
|
|
@pytest.fixture(autouse=True)
|
|
def setup():
|
|
setup_test_db(
|
|
User.__tablename__,
|
|
Package.__tablename__,
|
|
PackageBase.__tablename__,
|
|
PackageDependency.__tablename__,
|
|
PackageRelation.__tablename__,
|
|
PackageKeyword.__tablename__,
|
|
PackageVote.__tablename__,
|
|
PackageNotification.__tablename__,
|
|
PackageComaintainer.__tablename__,
|
|
OfficialProvider.__tablename__
|
|
)
|
|
|
|
|
|
@pytest.fixture
|
|
def client() -> TestClient:
|
|
""" Yield a FastAPI TestClient. """
|
|
yield TestClient(app=asgi.app)
|
|
|
|
|
|
@pytest.fixture
|
|
def user() -> User:
|
|
""" Yield a user. """
|
|
account_type = db.query(AccountType, AccountType.ID == USER_ID).first()
|
|
with db.begin():
|
|
user = db.create(User, Username="test",
|
|
Email="test@example.org",
|
|
Passwd="testPassword",
|
|
AccountType=account_type)
|
|
yield user
|
|
|
|
|
|
@pytest.fixture
|
|
def maintainer() -> User:
|
|
""" Yield a specific User used to maintain packages. """
|
|
account_type = db.query(AccountType, AccountType.ID == USER_ID).first()
|
|
with db.begin():
|
|
maintainer = db.create(User, Username="test_maintainer",
|
|
Email="test_maintainer@example.org",
|
|
Passwd="testPassword",
|
|
AccountType=account_type)
|
|
yield maintainer
|
|
|
|
|
|
@pytest.fixture
|
|
def package(maintainer: User) -> Package:
|
|
""" Yield a Package created by user. """
|
|
now = int(datetime.utcnow().timestamp())
|
|
with db.begin():
|
|
pkgbase = db.create(PackageBase,
|
|
Name="test-package",
|
|
Maintainer=maintainer,
|
|
Packager=maintainer,
|
|
Submitter=maintainer,
|
|
ModifiedTS=now)
|
|
package = db.create(Package,
|
|
PackageBase=pkgbase,
|
|
Name=pkgbase.Name)
|
|
yield package
|
|
|
|
|
|
@pytest.fixture
|
|
def packages(maintainer: User) -> List[Package]:
|
|
""" Yield 55 packages named pkg_0 .. pkg_54. """
|
|
packages_ = []
|
|
now = int(datetime.utcnow().timestamp())
|
|
with db.begin():
|
|
for i in range(55):
|
|
pkgbase = db.create(PackageBase,
|
|
Name=f"pkg_{i}",
|
|
Maintainer=maintainer,
|
|
Packager=maintainer,
|
|
Submitter=maintainer,
|
|
ModifiedTS=now)
|
|
package = db.create(Package,
|
|
PackageBase=pkgbase,
|
|
Name=f"pkg_{i}")
|
|
packages_.append(package)
|
|
|
|
yield packages_
|
|
|
|
|
|
def test_package_not_found(client: TestClient):
|
|
with client as request:
|
|
resp = request.get("/packages/not_found")
|
|
assert resp.status_code == int(HTTPStatus.NOT_FOUND)
|
|
|
|
|
|
def test_package_official_not_found(client: TestClient, package: Package):
|
|
""" When a Package has a matching OfficialProvider record, it is not
|
|
hosted on AUR, but in the official repositories. Getting a package
|
|
with this kind of record should return a status code 404. """
|
|
with db.begin():
|
|
db.create(OfficialProvider,
|
|
Name=package.Name,
|
|
Repo="core",
|
|
Provides=package.Name)
|
|
|
|
with client as request:
|
|
resp = request.get(package_endpoint(package))
|
|
assert resp.status_code == int(HTTPStatus.NOT_FOUND)
|
|
|
|
|
|
def test_package(client: TestClient, package: Package):
|
|
""" Test a single / packages / {name} route. """
|
|
with client as request:
|
|
|
|
resp = request.get(package_endpoint(package))
|
|
assert resp.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(resp.text)
|
|
h2 = root.find('.//div[@id="pkgdetails"]/h2')
|
|
|
|
sections = h2.text.split(":")
|
|
assert sections[0] == "Package Details"
|
|
|
|
name, version = sections[1].lstrip().split(" ")
|
|
assert name == package.Name
|
|
version == package.Version
|
|
|
|
rows = root.findall('.//table[@id="pkginfo"]//tr')
|
|
row = rows[1] # Second row is our target.
|
|
|
|
pkgbase = row.find("./td/a")
|
|
assert pkgbase.text.strip() == package.PackageBase.Name
|
|
|
|
|
|
def test_package_comments(client: TestClient, user: User, package: Package):
|
|
now = (datetime.utcnow().timestamp())
|
|
with db.begin():
|
|
comment = db.create(PackageComment, PackageBase=package.PackageBase,
|
|
User=user, Comments="Test comment", CommentTS=now)
|
|
|
|
cookies = {"AURSID": user.login(Request(), "testPassword")}
|
|
with client as request:
|
|
resp = request.get(package_endpoint(package), cookies=cookies)
|
|
assert resp.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(resp.text)
|
|
expected = [
|
|
comment.Comments
|
|
]
|
|
comments = root.xpath('.//div[contains(@class, "package-comments")]'
|
|
'/div[@class="article-content"]/div/text()')
|
|
for i, row in enumerate(expected):
|
|
assert comments[i].strip() == row
|
|
|
|
|
|
def test_package_requests_display(client: TestClient, user: User,
|
|
package: Package):
|
|
type_ = db.query(RequestType, RequestType.ID == DELETION_ID).first()
|
|
with db.begin():
|
|
db.create(PackageRequest, PackageBase=package.PackageBase,
|
|
PackageBaseName=package.PackageBase.Name,
|
|
User=user, RequestType=type_,
|
|
Comments="Test comment.",
|
|
ClosureComment=str())
|
|
|
|
# Test that a single request displays "1 pending request".
|
|
with client as request:
|
|
resp = request.get(package_endpoint(package))
|
|
assert resp.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(resp.text)
|
|
selector = '//div[@id="actionlist"]/ul/li/span[@class="flagged"]'
|
|
target = root.xpath(selector)[0]
|
|
assert target.text.strip() == "1 pending request"
|
|
|
|
type_ = db.query(RequestType, RequestType.ID == DELETION_ID).first()
|
|
with db.begin():
|
|
db.create(PackageRequest, PackageBase=package.PackageBase,
|
|
PackageBaseName=package.PackageBase.Name,
|
|
User=user, RequestType=type_,
|
|
Comments="Test comment2.",
|
|
ClosureComment=str())
|
|
|
|
# Test that a two requests display "2 pending requests".
|
|
with client as request:
|
|
resp = request.get(package_endpoint(package))
|
|
assert resp.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(resp.text)
|
|
selector = '//div[@id="actionlist"]/ul/li/span[@class="flagged"]'
|
|
target = root.xpath(selector)[0]
|
|
assert target.text.strip() == "2 pending requests"
|
|
|
|
|
|
def test_package_authenticated(client: TestClient, user: User,
|
|
package: Package):
|
|
""" We get the same here for either authenticated or not
|
|
authenticated. Form inputs are presented to maintainers.
|
|
This process also occurs when pkgbase.html is rendered. """
|
|
cookies = {"AURSID": user.login(Request(), "testPassword")}
|
|
with client as request:
|
|
resp = request.get(package_endpoint(package), cookies=cookies)
|
|
assert resp.status_code == int(HTTPStatus.OK)
|
|
|
|
expected = [
|
|
"View PKGBUILD",
|
|
"View Changes",
|
|
"Download snapshot",
|
|
"Search wiki",
|
|
"Flag package out-of-date",
|
|
"Vote for this package",
|
|
"Enable notifications",
|
|
"Submit Request"
|
|
]
|
|
for expected_text in expected:
|
|
assert expected_text in resp.text
|
|
|
|
# When no requests are up, make sure we don't see the display for them.
|
|
root = parse_root(resp.text)
|
|
selector = '//div[@id="actionlist"]/ul/li/span[@class="flagged"]'
|
|
target = root.xpath(selector)
|
|
assert len(target) == 0
|
|
|
|
|
|
def test_package_authenticated_maintainer(client: TestClient,
|
|
maintainer: User,
|
|
package: Package):
|
|
cookies = {"AURSID": maintainer.login(Request(), "testPassword")}
|
|
with client as request:
|
|
resp = request.get(package_endpoint(package), cookies=cookies)
|
|
assert resp.status_code == int(HTTPStatus.OK)
|
|
|
|
expected = [
|
|
"View PKGBUILD",
|
|
"View Changes",
|
|
"Download snapshot",
|
|
"Search wiki",
|
|
"Flag package out-of-date",
|
|
"Vote for this package",
|
|
"Enable notifications",
|
|
"Manage Co-Maintainers",
|
|
"Submit Request",
|
|
"Delete Package",
|
|
"Merge Package",
|
|
"Disown Package"
|
|
]
|
|
for expected_text in expected:
|
|
assert expected_text in resp.text
|
|
|
|
|
|
def test_package_dependencies(client: TestClient, maintainer: User,
|
|
package: Package):
|
|
# Create a normal dependency of type depends.
|
|
with db.begin():
|
|
dep_pkg = create_package("test-dep-1", maintainer)
|
|
dep = create_package_dep(package, dep_pkg.Name)
|
|
dep.DepArch = "x86_64"
|
|
|
|
# Also, create a makedepends.
|
|
make_dep_pkg = create_package("test-dep-2", maintainer)
|
|
make_dep = create_package_dep(package, make_dep_pkg.Name,
|
|
dep_type_name="makedepends")
|
|
|
|
# And... a checkdepends!
|
|
check_dep_pkg = create_package("test-dep-3", maintainer)
|
|
check_dep = create_package_dep(package, check_dep_pkg.Name,
|
|
dep_type_name="checkdepends")
|
|
|
|
# Geez. Just stop. This is optdepends.
|
|
opt_dep_pkg = create_package("test-dep-4", maintainer)
|
|
opt_dep = create_package_dep(package, opt_dep_pkg.Name,
|
|
dep_type_name="optdepends")
|
|
|
|
# Heh. Another optdepends to test one with a description.
|
|
opt_desc_dep_pkg = create_package("test-dep-5", maintainer)
|
|
opt_desc_dep = create_package_dep(package, opt_desc_dep_pkg.Name,
|
|
dep_type_name="optdepends")
|
|
opt_desc_dep.DepDesc = "Test description."
|
|
|
|
broken_dep = create_package_dep(package, "test-dep-6",
|
|
dep_type_name="depends")
|
|
|
|
# Create an official provider record.
|
|
db.create(OfficialProvider, Name="test-dep-99",
|
|
Repo="core", Provides="test-dep-99")
|
|
official_dep = create_package_dep(package, "test-dep-99")
|
|
|
|
# Also, create a provider who provides our test-dep-99.
|
|
provider = create_package("test-provider", maintainer)
|
|
create_package_rel(provider, dep.DepName)
|
|
|
|
with client as request:
|
|
resp = request.get(package_endpoint(package))
|
|
assert resp.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(resp.text)
|
|
|
|
expected = [
|
|
dep.DepName,
|
|
make_dep.DepName,
|
|
check_dep.DepName,
|
|
opt_dep.DepName,
|
|
opt_desc_dep.DepName,
|
|
official_dep.DepName
|
|
]
|
|
pkgdeps = root.findall('.//ul[@id="pkgdepslist"]/li/a')
|
|
for i, expectation in enumerate(expected):
|
|
assert pkgdeps[i].text.strip() == expectation
|
|
|
|
# Let's make sure the DepArch was displayed for our first dep.
|
|
arch = root.findall('.//ul[@id="pkgdepslist"]/li')[0]
|
|
arch = arch.xpath('./em')[1]
|
|
assert arch.text.strip() == "(x86_64)"
|
|
|
|
broken_node = root.find('.//ul[@id="pkgdepslist"]/li/span')
|
|
assert broken_node.text.strip() == broken_dep.DepName
|
|
|
|
|
|
def test_pkgbase_not_found(client: TestClient):
|
|
with client as request:
|
|
resp = request.get("/pkgbase/not_found")
|
|
assert resp.status_code == int(HTTPStatus.NOT_FOUND)
|
|
|
|
|
|
def test_pkgbase_redirect(client: TestClient, package: Package):
|
|
with client as request:
|
|
resp = request.get(f"/pkgbase/{package.Name}",
|
|
allow_redirects=False)
|
|
assert resp.status_code == int(HTTPStatus.SEE_OTHER)
|
|
assert resp.headers.get("location") == f"/packages/{package.Name}"
|
|
|
|
|
|
def test_pkgbase(client: TestClient, package: Package):
|
|
with db.begin():
|
|
second = db.create(Package, Name="second-pkg",
|
|
PackageBase=package.PackageBase)
|
|
|
|
expected = [package.Name, second.Name]
|
|
with client as request:
|
|
resp = request.get(f"/pkgbase/{package.Name}",
|
|
allow_redirects=False)
|
|
assert resp.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(resp.text)
|
|
|
|
# Check the details box title.
|
|
title = root.find('.//div[@id="pkgdetails"]/h2')
|
|
title, pkgname = title.text.split(": ")
|
|
assert title == "Package Base Details"
|
|
assert pkgname == package.Name
|
|
|
|
pkgs = root.findall('.//div[@id="pkgs"]/ul/li/a')
|
|
for i, name in enumerate(expected):
|
|
assert pkgs[i].text.strip() == name
|
|
|
|
|
|
def test_packages(client: TestClient, packages: List[Package]):
|
|
""" Test the / packages route with defaults.
|
|
|
|
Defaults:
|
|
50 results per page
|
|
offset of 0
|
|
"""
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SeB": "X" # "X" isn't valid, defaults to "nd"
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(response.text)
|
|
stats = root.xpath('//div[@class="pkglist-stats"]/p')[0]
|
|
pager_text = re.sub(r'\s+', " ", stats.text.replace("\n", "").strip())
|
|
assert pager_text == "55 packages found. Page 1 of 2."
|
|
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
assert len(rows) == 50 # Default per-page
|
|
|
|
|
|
def test_packages_search_by_name(client: TestClient, packages: List[Package]):
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SeB": "n",
|
|
"K": "pkg_"
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(response.text)
|
|
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
assert len(rows) == 50 # Default per-page
|
|
|
|
|
|
def test_packages_search_by_exact_name(client: TestClient,
|
|
packages: List[Package]):
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SeB": "N",
|
|
"K": "pkg_"
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
|
|
# There is no package named exactly 'pkg_', we get 0 results.
|
|
assert len(rows) == 0
|
|
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SeB": "N",
|
|
"K": "pkg_1"
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
|
|
# There's just one package named 'pkg_1', we get 1 result.
|
|
assert len(rows) == 1
|
|
|
|
|
|
def test_packages_search_by_pkgbase(client: TestClient,
|
|
packages: List[Package]):
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SeB": "b",
|
|
"K": "pkg_"
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(response.text)
|
|
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
assert len(rows) == 50
|
|
|
|
|
|
def test_packages_search_by_exact_pkgbase(client: TestClient,
|
|
packages: List[Package]):
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SeB": "B",
|
|
"K": "pkg_"
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
assert len(rows) == 0
|
|
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SeB": "B",
|
|
"K": "pkg_1"
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
assert len(rows) == 1
|
|
|
|
|
|
def test_packages_search_by_keywords(client: TestClient,
|
|
packages: List[Package]):
|
|
# None of our packages have keywords, so this query should return nothing.
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SeB": "k",
|
|
"K": "testKeyword"
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
assert len(rows) == 0
|
|
|
|
# But now, let's create the keyword for the first package.
|
|
package = packages[0]
|
|
with db.begin():
|
|
db.create(PackageKeyword,
|
|
PackageBase=package.PackageBase,
|
|
Keyword="testKeyword")
|
|
|
|
# And request packages with that keyword, we should get 1 result.
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SeB": "k",
|
|
"K": "testKeyword"
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
assert len(rows) == 1
|
|
|
|
|
|
def test_packages_search_by_maintainer(client: TestClient,
|
|
maintainer: User,
|
|
package: Package):
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SeB": "m",
|
|
"K": maintainer.Username
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
assert len(rows) == 1
|
|
|
|
|
|
def test_packages_search_by_comaintainer(client: TestClient,
|
|
maintainer: User,
|
|
package: Package):
|
|
# Nobody's a comaintainer yet.
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SeB": "c",
|
|
"K": maintainer.Username
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
assert len(rows) == 0
|
|
|
|
# Now, we create a comaintainer.
|
|
with db.begin():
|
|
db.create(PackageComaintainer,
|
|
PackageBase=package.PackageBase,
|
|
User=maintainer,
|
|
Priority=1)
|
|
|
|
# Then test that it's returned by our search.
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SeB": "c",
|
|
"K": maintainer.Username
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
assert len(rows) == 1
|
|
|
|
|
|
def test_packages_search_by_co_or_maintainer(client: TestClient,
|
|
maintainer: User,
|
|
package: Package):
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SeB": "M",
|
|
"SB": "BLAH", # Invalid SB; gets reset to default "n".
|
|
"K": maintainer.Username
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
assert len(rows) == 1
|
|
|
|
with db.begin():
|
|
user = db.create(User, Username="comaintainer",
|
|
Email="comaintainer@example.org",
|
|
Passwd="testPassword")
|
|
db.create(PackageComaintainer,
|
|
PackageBase=package.PackageBase,
|
|
User=user,
|
|
Priority=1)
|
|
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SeB": "M",
|
|
"K": user.Username
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
assert len(rows) == 1
|
|
|
|
|
|
def test_packages_search_by_submitter(client: TestClient,
|
|
maintainer: User,
|
|
package: Package):
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SeB": "s",
|
|
"K": maintainer.Username
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
assert len(rows) == 1
|
|
|
|
|
|
def test_packages_sort_by_votes(client: TestClient,
|
|
maintainer: User,
|
|
packages: List[Package]):
|
|
# Set the first package's NumVotes to 1.
|
|
with db.begin():
|
|
packages[0].PackageBase.NumVotes = 1
|
|
|
|
# Test that, by default, the first result is what we just set above.
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SB": "v" # Votes.
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
votes = rows[0].xpath('./td')[2] # The third column of the first row.
|
|
assert votes.text.strip() == "1"
|
|
|
|
# Now, test that with an ascending order, the last result is
|
|
# the one we set, since the default (above) is descending.
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SB": "v", # Votes.
|
|
"SO": "a", # Ascending.
|
|
"O": "50" # Second page.
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
votes = rows[-1].xpath('./td')[2] # The third column of the last row.
|
|
assert votes.text.strip() == "1"
|
|
|
|
|
|
def test_packages_sort_by_popularity(client: TestClient,
|
|
maintainer: User,
|
|
packages: List[Package]):
|
|
# Set the first package's Popularity to 0.50.
|
|
with db.begin():
|
|
packages[0].PackageBase.Popularity = "0.50"
|
|
|
|
# Test that, by default, the first result is what we just set above.
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SB": "p" # Popularity
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
pop = rows[0].xpath('./td')[3] # The fourth column of the first row.
|
|
assert pop.text.strip() == "0.50"
|
|
|
|
|
|
def test_packages_sort_by_voted(client: TestClient,
|
|
maintainer: User,
|
|
packages: List[Package]):
|
|
now = int(datetime.utcnow().timestamp())
|
|
with db.begin():
|
|
db.create(PackageVote, PackageBase=packages[0].PackageBase,
|
|
User=maintainer, VoteTS=now)
|
|
|
|
# Test that, by default, the first result is what we just set above.
|
|
cookies = {"AURSID": maintainer.login(Request(), "testPassword")}
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SB": "w", # Voted
|
|
"SO": "d" # Descending, Voted first.
|
|
}, cookies=cookies)
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
voted = rows[0].xpath('./td')[5] # The sixth column of the first row.
|
|
assert voted.text.strip() == "Yes"
|
|
|
|
# Conversely, everything else was not voted on.
|
|
voted = rows[1].xpath('./td')[5] # The sixth column of the second row.
|
|
assert voted.text.strip() == str() # Empty.
|
|
|
|
|
|
def test_packages_sort_by_notify(client: TestClient,
|
|
maintainer: User,
|
|
packages: List[Package]):
|
|
db.create(PackageNotification,
|
|
PackageBase=packages[0].PackageBase,
|
|
User=maintainer)
|
|
|
|
# Test that, by default, the first result is what we just set above.
|
|
cookies = {"AURSID": maintainer.login(Request(), "testPassword")}
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SB": "o", # Voted
|
|
"SO": "d" # Descending, Voted first.
|
|
}, cookies=cookies)
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
notify = rows[0].xpath('./td')[6] # The sixth column of the first row.
|
|
assert notify.text.strip() == "Yes"
|
|
|
|
# Conversely, everything else was not voted on.
|
|
notify = rows[1].xpath('./td')[6] # The sixth column of the second row.
|
|
assert notify.text.strip() == str() # Empty.
|
|
|
|
|
|
def test_packages_sort_by_maintainer(client: TestClient,
|
|
maintainer: User,
|
|
package: Package):
|
|
""" Sort a package search by the maintainer column. """
|
|
|
|
# Create a second package, so the two can be ordered and checked.
|
|
with db.begin():
|
|
maintainer2 = db.create(User, Username="maintainer2",
|
|
Email="maintainer2@example.org",
|
|
Passwd="testPassword")
|
|
base2 = db.create(PackageBase, Name="pkg_2", Maintainer=maintainer2,
|
|
Submitter=maintainer2, Packager=maintainer2)
|
|
db.create(Package, Name="pkg_2", PackageBase=base2)
|
|
|
|
# Check the descending order route.
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SB": "m",
|
|
"SO": "d"
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
col = rows[0].xpath('./td')[5].xpath('./a')[0] # Last column.
|
|
|
|
assert col.text.strip() == maintainer.Username
|
|
|
|
# On the other hand, with ascending, we should get reverse ordering.
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SB": "m",
|
|
"SO": "a"
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
col = rows[0].xpath('./td')[5].xpath('./a')[0] # Last column.
|
|
|
|
assert col.text.strip() == maintainer2.Username
|
|
|
|
|
|
def test_packages_sort_by_last_modified(client: TestClient,
|
|
packages: List[Package]):
|
|
now = int(datetime.utcnow().timestamp())
|
|
# Set the first package's ModifiedTS to be 1000 seconds before now.
|
|
package = packages[0]
|
|
with db.begin():
|
|
package.PackageBase.ModifiedTS = now - 1000
|
|
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"SB": "l",
|
|
"SO": "a" # Ascending; oldest modification first.
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
# We should have 50 (default per page) results.
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
assert len(rows) == 50
|
|
|
|
# Let's assert that the first item returned was the one we modified above.
|
|
row = rows[0]
|
|
col = row.xpath('./td')[0].xpath('./a')[0]
|
|
assert col.text.strip() == package.Name
|
|
|
|
|
|
def test_packages_flagged(client: TestClient, maintainer: User,
|
|
packages: List[Package]):
|
|
package = packages[0]
|
|
|
|
now = int(datetime.utcnow().timestamp())
|
|
|
|
with db.begin():
|
|
package.PackageBase.OutOfDateTS = now
|
|
package.PackageBase.Flagger = maintainer
|
|
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"outdated": "on"
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
# We should only get one result from this query; the package we flagged.
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
assert len(rows) == 1
|
|
|
|
with client as request:
|
|
response = request.get("/packages", params={
|
|
"outdated": "off"
|
|
})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
# In this case, we should get 54 results, which means that the first
|
|
# page will have 50 results (55 packages - 1 outdated = 54 not outdated).
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
assert len(rows) == 50
|
|
|
|
|
|
def test_packages_orphans(client: TestClient, packages: List[Package]):
|
|
package = packages[0]
|
|
with db.begin():
|
|
package.PackageBase.Maintainer = None
|
|
|
|
with client as request:
|
|
response = request.get("/packages", params={"submit": "Orphans"})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
|
|
# We only have one orphan. Let's make sure that's what is returned.
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
assert len(rows) == 1
|
|
|
|
|
|
def test_packages_per_page(client: TestClient, maintainer: User):
|
|
""" Test the ability for /packages to deal with the PP query
|
|
argument specifications (50, 100, 250; default: 50). """
|
|
with db.begin():
|
|
for i in range(255):
|
|
base = db.create(PackageBase, Name=f"pkg_{i}",
|
|
Maintainer=maintainer,
|
|
Submitter=maintainer,
|
|
Packager=maintainer)
|
|
db.create(Package, PackageBase=base, Name=base.Name)
|
|
|
|
# Test default case, PP of 50.
|
|
with client as request:
|
|
response = request.get("/packages", params={"PP": 50})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
assert len(rows) == 50
|
|
|
|
# Alright, test the next case, PP of 100.
|
|
with client as request:
|
|
response = request.get("/packages", params={"PP": 100})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
assert len(rows) == 100
|
|
|
|
# And finally, the last case, a PP of 250.
|
|
with client as request:
|
|
response = request.get("/packages", params={"PP": 250})
|
|
assert response.status_code == int(HTTPStatus.OK)
|
|
root = parse_root(response.text)
|
|
rows = root.xpath('//table[@class="results"]/tbody/tr')
|
|
assert len(rows) == 250
|