Manage ASL magazines and their articles.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
asl-articles/asl_articles/tests/test_articles.py

837 lines
30 KiB

""" Test article operations. """
import os
import urllib.request
import urllib.error
import json
import base64
import re
from asl_articles.search import SEARCH_ALL_ARTICLES
from asl_articles.tests.utils import init_tests, select_main_menu_option, select_sr_menu_option, \
do_search, get_search_results, find_search_result, get_search_result_names, check_search_result, \
do_test_confirm_discard_changes, find_parent_by_class, \
wait_for, wait_for_elem, wait_for_not_elem, find_child, find_children, \
set_elem_text, set_toast_marker, check_toast, send_upload_data, change_image, remove_image, get_article_row, \
check_ask_dialog, check_error_msg, check_constraint_warnings
from asl_articles.tests.react_select import ReactSelect
# ---------------------------------------------------------------------
def test_edit_article( webdriver, flask_app, dbconn ):
"""Test editing articles."""
# initialize
init_tests( webdriver, flask_app, dbconn, fixtures="articles.json" )
# edit an article
results = do_search( SEARCH_ALL_ARTICLES )
sr = find_search_result( "What To Do If You Have A Tin Can", results )
edit_article( sr, {
"title": " Updated title ",
"subtitle": " Updated subtitle ",
"snippet": " Updated snippet. ",
"pageno": " 123 ",
"authors": [ "+Fred Nerk", "+Joe Blow" ],
"tags": [ "+abc", "+xyz" ],
"url": " http://updated-article.com ",
} )
# check that the search result was updated in the UI
sr = check_search_result( "Updated title", _check_sr, [
"Updated title", "Updated subtitle", "Updated snippet.", "123",
[ "Fred Nerk", "Joe Blow" ],
[ "abc", "xyz" ],
"http://updated-article.com/"
] )
# remove all fields from the article
edit_article( sr, {
"title": "Tin Cans Rock!",
"subtitle": "",
"snippet": "",
"pageno": "",
"authors": [ "-Fred Nerk", "-Joe Blow" ],
"tags": [ "-abc", "-xyz" ],
"url": "",
} )
# check that the search result was updated in the UI
expected = [ "Tin Cans Rock!", None, "", "", [], [], None ]
check_search_result( expected[0], _check_sr, expected )
# check that the article was updated in the database
do_search( SEARCH_ALL_ARTICLES )
check_search_result( expected[0], _check_sr, expected )
# ---------------------------------------------------------------------
def test_create_article( webdriver, flask_app, dbconn ):
"""Test creating new articles."""
# initialize
init_tests( webdriver, flask_app, dbconn, fixtures="articles.json" )
do_search( SEARCH_ALL_ARTICLES )
# create a new article
create_article( {
"title": "New article",
"subtitle": "New subtitle",
"snippet": "New snippet.",
"pageno": "99",
"authors": [ "+Me" ],
"tags": [ "+111", "+222", "+333" ],
"url": "http://new-snippet.com"
} )
# check that the new article appears in the UI
expected = [
"New article", "New subtitle", "New snippet.", "99",
[ "Me" ],
[ "111", "222", "333" ],
"http://new-snippet.com/"
]
check_search_result( expected[0], _check_sr, expected )
# check that the new article has been saved in the database
do_search( SEARCH_ALL_ARTICLES )
check_search_result( expected[0], _check_sr, expected )
# ---------------------------------------------------------------------
def test_constraints( webdriver, flask_app, dbconn ):
"""Test constraint validation."""
# initialize
init_tests( webdriver, flask_app, dbconn, disable_constraints=False, fixtures="publications.json" )
# try to create an article with no title
dlg = create_article( {}, expected_error="Please give it a title." )
def do_create_test( vals, expected ):
return create_article( vals, dlg=dlg, expected_constraints=expected )
def do_edit_test( sr, vals, expected ):
return edit_article( sr, vals, expected_constraints=expected )
# set the article's title
do_create_test( { "title": "New article" }, [
"No publication was specified.",
"No snippet was provided.",
"No authors were specified."
] )
# set the article's page number
do_create_test( { "pageno": 99 }, [
"No publication was specified.",
"A page number was specified but no publication.",
"No snippet was provided.",
"No authors were specified."
] )
# assign the article to a publisher
do_create_test( { "publication": "MMP News", "pageno": "" }, [
"No page number was specified.",
"No snippet was provided.",
"No authors were specified."
] )
# set a non-numeric page number
do_create_test( { "pageno": "foo!" }, [
"The page number is not numeric.",
"No snippet was provided.",
"No authors were specified."
] )
# set the article's page number and provide a snippet
do_create_test( { "pageno": 123, "snippet": "Article snippet." }, [
"No authors were specified."
] )
# accept the constraint warnings
find_child( "button.ok", dlg ).click()
find_child( "#ask button.ok" ).click()
results = wait_for( 2, get_search_results )
article_sr = results[0]
# check that the search result was updated in the UI
check_search_result( article_sr, _check_sr, [
"New article", "", "Article snippet.", "123", [], [], None
] )
# try editing the article
dlg = do_edit_test( article_sr, {}, [
"No authors were specified."
] )
find_child( "button.cancel", dlg ).click()
# set the article's author
do_edit_test( article_sr, { "authors": ["+Joe Blow"] }, None )
# check that the search result was updated in the UI
check_search_result( article_sr, _check_sr, [
"New article", "", "Article snippet.", "123", ["Joe Blow"], [], None
] )
# ---------------------------------------------------------------------
def test_confirm_discard_changes( webdriver, flask_app, dbconn ):
"""Test confirmation of discarding changes made to a dialog."""
# initialize
init_tests( webdriver, flask_app, dbconn, disable_confirm_discard_changes=False, fixtures="publications.json" )
# do the test
def update_react_select( elem, val ):
select = ReactSelect( find_parent_by_class( elem, "react-select" ) )
select.select_by_name( val )
def update_multiselect( elem, vals ):
select = ReactSelect( find_parent_by_class( elem, "react-select" ) )
select.update_multiselect_values( *vals )
do_test_confirm_discard_changes( "new-article", {
"publication": (
lambda elem: update_react_select( elem, "MMP News" ),
lambda elem: update_react_select( elem, "(none)" )
),
"authors": (
lambda elem: update_multiselect( elem, ["+Joe Blow"] ),
lambda elem: update_multiselect( elem, ["-Joe Blow"] ),
),
"scenarios": (
lambda elem: update_multiselect( elem, ["+Hill 621 [E]"] ),
lambda elem: update_multiselect( elem, ["-Hill 621 [E]"] ),
),
"tags": (
lambda elem: update_multiselect( elem, ["+foo"] ),
lambda elem: update_multiselect( elem, ["-foo"] ),
)
} )
# ---------------------------------------------------------------------
def test_delete_article( webdriver, flask_app, dbconn ):
"""Test deleting articles."""
# initialize
init_tests( webdriver, flask_app, dbconn, fixtures="articles.json" )
# start to delete an article, but cancel the operation
article_name = "Smoke Gets In Your Eyes"
results = do_search( SEARCH_ALL_ARTICLES )
result_names = get_search_result_names( results )
sr = find_search_result( article_name, results )
select_sr_menu_option( sr, "delete" )
check_ask_dialog( ( "Delete this article?", article_name ), "cancel" )
# check that search results are unchanged on-screen
results2 = get_search_results()
assert results2 == results
# check that the search results are unchanged in the database
results3 = do_search( SEARCH_ALL_ARTICLES )
assert get_search_result_names( results3 ) == result_names
# delete the article
sr = find_search_result( article_name, results3 )
select_sr_menu_option( sr, "delete" )
set_toast_marker( "info" )
check_ask_dialog( ( "Delete this article?", article_name ), "ok" )
wait_for( 2, lambda: check_toast( "info", "The article was deleted." ) )
# check that search result was removed on-screen
wait_for( 2, lambda: article_name not in get_search_result_names() )
# check that the search result was deleted from the database
results = do_search( SEARCH_ALL_ARTICLES )
assert article_name not in get_search_result_names( results )
# ---------------------------------------------------------------------
def test_images( webdriver, flask_app, dbconn ): #pylint: disable=too-many-statements
"""Test article images."""
# initialize
init_tests( webdriver, flask_app, dbconn, max_image_upload_size=2*1024 )
article_sr = article_id = None
def check_image( expected ):
# check the image in the article's search result
def check_sr_image():
img = find_child( "img.image", article_sr )
if expected:
expected_url = flask_app.url_for( "get_image", image_type="article", image_id=article_id )
image_url = img.get_attribute( "src" ).split( "?" )[0]
return image_url == expected_url
else:
return not img
wait_for( 2, check_sr_image )
# check the image in the article's config
select_sr_menu_option( article_sr, "edit" )
dlg = wait_for_elem( 2, "#article-form" )
if expected:
# make sure there is an image
img = find_child( ".row.image img.image", dlg )
image_url = img.get_attribute( "src" )
assert "/images/article/{}".format( article_id ) in image_url
# make sure the "remove image" icon is visible
btn = find_child( ".row.image .remove-image", dlg )
assert btn.is_displayed()
# make sure the article's image is correct
resp = urllib.request.urlopen( image_url ).read()
assert resp == open( expected, "rb" ).read()
else:
# make sure there is no image
img = find_child( ".row.image img.image", dlg )
assert img.get_attribute( "src" ).endswith( "/images/placeholder.png" )
# make sure the "remove image" icon is hidden
btn = find_child( ".row.image .remove-image", dlg )
assert not btn.is_displayed()
# make sure the article's image is not available
url = flask_app.url_for( "get_image", image_type="article", image_id=article_id )
try:
resp = urllib.request.urlopen( url )
assert False, "Should never get here!"
except urllib.error.HTTPError as ex:
assert ex.code == 404
find_child( ".cancel", dlg ).click()
# create an article with no image
create_article( { "title": "Test Article" } )
results = get_search_results()
assert len(results) == 1
article_sr = results[0]
article_id = article_sr.get_attribute( "testing--article_id" )
check_image( None )
# add an image to the article
fname = os.path.join( os.path.split(__file__)[0], "fixtures/images/1.gif" )
edit_article( article_sr, { "image": fname } )
check_image( fname )
# change the article's image
fname = os.path.join( os.path.split(__file__)[0], "fixtures/images/2.gif" )
edit_article( article_sr, { "image": fname } )
check_image( fname )
# remove the article's image
edit_article( article_sr, { "image": None } )
check_image( None )
# try to upload an image that's too large
select_sr_menu_option( article_sr, "edit" )
dlg = wait_for_elem( 2, "#article-form" )
data = base64.b64encode( 5000 * b" " )
data = "{}|{}".format( "too-big.png", data.decode("ascii") )
send_upload_data( data,
lambda: find_child( ".row.image img.image", dlg ).click()
)
check_error_msg( "The file must be no more than 2 KB in size." )
# ---------------------------------------------------------------------
def test_parent_publisher( webdriver, flask_app, dbconn ):
"""Test setting an article's parent publication."""
# initialize
init_tests( webdriver, flask_app, dbconn, fixtures="parents.json" )
def check_result( sr, expected_parent ): #pylint: disable=too-many-return-statements
# check that the parent publication was updated in the UI
elem = find_child( ".header .publication", sr )
if expected_parent:
if elem.text != "{}".format( expected_parent[1] ):
return None
else:
if elem is not None:
return None
# check that the parent publication was updated in the database
article_id = sr.get_attribute( "testing--article_id" )
url = flask_app.url_for( "get_article", article_id=article_id )
article = json.load( urllib.request.urlopen( url ) )
if expected_parent:
if article["pub_id"] != expected_parent[0]:
return None
else:
if article["pub_id"] is not None:
return None
# check that the parent publication was updated in the UI
results = do_search( '"My Article"' )
assert len(results) == 1
sr = results[0]
elem = find_child( ".header .publication", sr )
if expected_parent:
if elem.text != "{}".format( expected_parent[1] ):
return None
else:
if elem is not None:
return None
return sr
# create an article with no parent publication
create_article( { "title": "My Article" } )
results = get_search_results()
assert len(results) == 1
sr = wait_for( 2, lambda: check_result( results[0], None ) )
# change the article to have a publication
edit_article( sr, { "publication": "ASL Journal" } )
sr = wait_for( 2, lambda: check_result( sr, (1, "ASL Journal") ) )
# change the article back to having no publication
edit_article( sr, { "publication": "(none)" } )
sr = wait_for( 2, lambda: check_result( sr, None ) )
# ---------------------------------------------------------------------
def test_publisher_articles( webdriver, flask_app, dbconn ): #pylint: disable=too-many-statements
"""Test articles that are associated with a publisher (not publication)."""
# initialize
init_tests( webdriver, flask_app, dbconn, fixtures="publisher-articles.json" )
def check_parent_in_sr( sr, pub, publ ):
"""Check the article's parent publication/publisher in a search result."""
if pub:
elem = wait_for( 2, lambda: find_child( ".header a.publication", sr ) )
assert elem.is_displayed()
assert elem.text == pub
assert re.search( r"^http://.+?/publication/\d+", elem.get_attribute( "href" ) )
elif publ:
elem = wait_for( 2, lambda: find_child( ".header a.publisher", sr ) )
assert elem.is_displayed()
assert elem.text == publ
assert re.search( r"^http://.+?/publisher/\d+", elem.get_attribute( "href" ) )
else:
assert False, "At least one publication/publisher must be specified."
def check_parent_in_dlg( dlg, pub, publ ):
"""Check the article's parent publication/publication in the edit dialog."""
if pub:
select = find_child( ".row.publication .react-select", dlg )
assert select.is_displayed()
assert select.text == pub
elif publ:
select = find_child( ".row.publisher .react-select", dlg )
assert select.is_displayed()
assert select.text == publ
else:
assert False, "At least one publication/publisher must be specified."
# create an article associated with LFT
create_article( {
"title": "test article",
"publisher": "Le Franc Tireur"
} )
results = wait_for( 2, get_search_results )
assert len(results) == 1
sr = results[0]
check_parent_in_sr( sr, None, "Le Franc Tireur" )
# open the article's dialog
select_sr_menu_option( sr, "edit" )
dlg = wait_for_elem( 2, "#article-form" )
check_parent_in_dlg( dlg, None, "Le Franc Tireur" )
# change the article to be associated with an MMP publication
find_child( ".row.publisher label.parent-mode" ).click()
select = wait_for_elem( 2, ".row.publication .react-select" )
ReactSelect( select ).select_by_name( "MMP News" )
find_child( "button.ok", dlg ).click()
results = wait_for( 2, get_search_results )
assert len(results) == 1
sr = results[0]
check_parent_in_sr( sr, "MMP News", None )
# open the article's dialog
select_sr_menu_option( sr, "edit" )
dlg = wait_for_elem( 2, "#article-form" )
check_parent_in_dlg( dlg, "MMP News", None )
# change the article to be associated with MMP (publisher)
find_child( ".row.publication label.parent-mode" ).click()
select = wait_for_elem( 2, ".row.publisher .react-select" )
ReactSelect( select ).select_by_name( "Multiman Publishing" )
find_child( "button.ok", dlg ).click()
results = wait_for( 2, get_search_results )
assert len(results) == 1
sr = results[0]
check_parent_in_sr( sr, None, "Multiman Publishing" )
# show the MMP publisher
results = do_search( "multiman" )
assert len(results) == 1
sr = results[0]
collapsibles = find_children( ".collapsible", sr )
assert len(collapsibles) == 2
items = find_children( "li a", collapsibles[1] )
assert len(items) == 1
item = items[0]
assert item.text == "test article"
assert re.search( r"^http://.+?/article/\d+", item.get_attribute( "href" ) )
# delete the MMP publisher
# NOTE: There are 2 MMP articles, the one that is in the "MMP News" publication,
# and the test article we created above that is associated with the publisher.
select_sr_menu_option( sr, "delete" )
check_ask_dialog( ( "Delete this publisher?", "2 articles will also be deleted" ), "ok" )
query = dbconn.execute( "SELECT count(*) FROM article" )
assert query.scalar() == 0
# ---------------------------------------------------------------------
def test_publisher_article_dates( webdriver, flask_app, dbconn ):
"""Test "published" dates for publisher articles."""
# initialize
init_tests( webdriver, flask_app, dbconn, disable_constraints=False, fixtures="publisher-article-dates.json" )
# initialize
article_title, article_date = "test article", "1st January, 2000"
article_sr = None
def check_article_date( has_date ):
# check the article's publication date
def do_check():
elem = find_child( ".article_date", article_sr )
article_id = article_sr.get_attribute( "testing--article_id" )
row = get_article_row( dbconn, article_id, ["article_date"] )
if has_date:
return elem.text == article_date and row[0] == article_date
else:
return not elem and not row[0]
wait_for( 2, do_check )
# create an article associated with a publication
create_article( {
"title": article_title,
"publication": "ASL Journal",
"snippet": "This is a test article.",
"pageno": 42,
"authors": [ "+Joe Blow" ]
} )
article_sr = wait_for( 2, lambda: find_search_result( article_title ) )
check_article_date( False )
# change the article to be associated with a publisher
edit_article( article_sr, {
"publisher": "Avalon Hill"
}, expected_constraints = [
"The article date was not specified."
], accept_constraints=True )
check_article_date( False )
# give the article a published date
edit_article( article_sr, {
"article_date": article_date
} )
check_article_date( True )
# change the article back to the publication
edit_article( article_sr, {
"publication": "ASL Journal"
} )
check_article_date( False )
# ---------------------------------------------------------------------
def test_unicode( webdriver, flask_app, dbconn ):
"""Test Unicode content."""
# initialize
init_tests( webdriver, flask_app, dbconn )
# create a article with Unicode content
create_article( {
"title": "japan = \u65e5\u672c",
"subtitle": "s.korea = \ud55c\uad6d",
"snippet": "greece = \u0395\u03bb\u03bb\u03ac\u03b4\u03b1",
"tags": [ "+\u0e51", "+\u0e52", "+\u0e53" ],
"url": "http://\ud55c\uad6d.com"
} )
# check that the new article is showing the Unicode content correctly
results = do_search( SEARCH_ALL_ARTICLES )
assert len(results) == 1
check_search_result( results[0], _check_sr, [
"japan = \u65e5\u672c",
"s.korea = \ud55c\uad6d",
"greece = \u0395\u03bb\u03bb\u03ac\u03b4\u03b1",
"",
[],
[ "\u0e51", "\u0e52", "\u0e53" ],
"http://xn--3e0b707e.com/"
] )
# ---------------------------------------------------------------------
def test_clean_html( webdriver, flask_app, dbconn ):
"""Test cleaning HTML content."""
# initialize
init_tests( webdriver, flask_app, dbconn )
replace = [
"[\u00ab\u00bb\u201c\u201d\u201e\u201f foo\u2014bar \u2018\u2019\u201a\u201b\u2039\u203a]",
"[\"\"\"\"\"\" foo - bar '''''']"
]
# create a article with HTML content
create_article( {
"title": "title: <span onclick='boo!'> <b>bold</b> <xxx>xxx</xxx> <i>italic</i> {}".format( replace[0] ),
"subtitle": "<i>italicized subtitle</i> {}".format( replace[0] ),
"snippet": "bad stuff here: <script>HCF</script> {}".format( replace[0] )
}, toast_type="warning" )
# check that the HTML was cleaned
sr = check_search_result( None, _check_sr, [
"title: bold xxx italic {}".format( replace[1] ),
"italicized subtitle {}".format( replace[1] ),
"bad stuff here: {}".format( replace[1] ),
"", [], [], None
] )
assert find_child( ".title", sr ).get_attribute( "innerHTML" ) \
== "title: <span> <b>bold</b> xxx <i>italic</i> {}</span>".format( replace[1] )
assert find_child( ".subtitle", sr ).get_attribute( "innerHTML" ) \
== "<i>italicized subtitle</i> {}".format( replace[1] )
assert check_toast( "warning", "Some values had HTML cleaned up.", contains=True )
# update the article with new HTML content
edit_article( sr, {
"title": "<div onclick='...'>updated</div>"
}, toast_type="warning" )
wait_for( 2, lambda: get_search_result_names() == ["updated"] )
assert check_toast( "warning", "Some values had HTML cleaned up.", contains=True )
# ---------------------------------------------------------------------
def test_timestamps( webdriver, flask_app, dbconn ):
"""Test setting of timestamps."""
# initialize
init_tests( webdriver, flask_app, dbconn )
# create an article
create_article( { "title": "My Article" } )
results = get_search_results()
assert len(results) == 1
article_sr = results[0]
article_id = article_sr.get_attribute( "testing--article_id" )
# check its timestamps
row = get_article_row( dbconn, article_id, ["time_created","time_updated"] )
assert row[0]
assert row[1] is None
# update the article
edit_article( article_sr, { "title": "My Article (updated)" } )
# check its timestamps
row2 = get_article_row( dbconn, article_id, ["time_created","time_updated"] )
assert row2[0] == row[0]
assert row2[1] > row2[0]
# ---------------------------------------------------------------------
def test_article_ratings( webdriver, flask_app, dbconn ):
"""Test article ratings."""
# initialize
init_tests( webdriver, flask_app, dbconn, fixtures="articles.json" )
def do_test( article_sr, star_no, expected ):
# click the specified article star
stars = find_children( ".rating-stars img", article_sr )
stars[ star_no ].click()
for sr_no,sr in enumerate(results):
assert get_rating(sr) == expected[sr_no]
# compare the ratings on-screen with what's in the database
for sr in results:
article_id = sr.get_attribute( "testing--article_id" )
ui_rating = get_rating( sr )
db_rating = dbconn.execute(
"SELECT article_rating FROM article WHERE article_id={}".format( article_id )
).scalar()
if db_rating is None:
assert ui_rating == 0
else:
assert ui_rating == db_rating
def get_rating( article_sr ):
stars = [
"disabled" not in star.get_attribute("src")
for star in find_children( ".rating-stars img", article_sr )
]
rating = 0
for star in stars:
if not star:
assert all( not s for s in stars[rating+1:] )
break
rating += 1
return rating
# get the test articles
results = do_search( SEARCH_ALL_ARTICLES )
# do the tests
do_test( results[0], 2, [3,0] )
do_test( results[1], 1, [3,2] )
# do the tests
do_test( results[0], 2, [2,2] )
do_test( results[0], 2, [3,2] )
do_test( results[0], 0, [1,2] )
do_test( results[0], 0, [0,2] )
do_test( results[0], 0, [1,2] )
# ---------------------------------------------------------------------
def create_article( vals, toast_type="info",
expected_error=None, expected_constraints=None, accept_constraints=False,
dlg=None
):
"""Create a new article."""
# initialize
set_toast_marker( toast_type )
# create the new article
if not dlg:
select_main_menu_option( "new-article" )
dlg = wait_for_elem( 2, "#article-form" )
_update_values( dlg, vals )
find_child( "button.ok", dlg ).click()
# check what happened
if expected_error:
# we were expecting an error, confirm the error message
check_error_msg( expected_error )
return dlg # nb: the dialog is left on-screen
elif expected_constraints:
# we were expecting constraint warnings, confirm them
check_constraint_warnings( "Do you want to create this article?",
expected_constraints, "ok" if accept_constraints else "cancel"
)
return dlg # nb: the dialog is left on-screen
else:
# we were expecting the create to work, confirm this
wait_for( 2,
lambda: check_toast( toast_type, "created OK", contains=True )
)
wait_for_not_elem( 2, "#article-form" )
return None
# - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
def edit_article( sr, vals, toast_type="info",
expected_error=None, expected_constraints=None, accept_constraints=False
): #pylint: disable=too-many-branches
"""Edit a article's details."""
# initialize
if sr:
select_sr_menu_option( sr, "edit" )
else:
pass # nb: we assume that the dialog is already on-screen
dlg = wait_for_elem( 2, "#article-form" )
# update the specified article's details
_update_values( dlg, vals )
set_toast_marker( toast_type )
find_child( "button.ok", dlg ).click()
# check what happened
if expected_error:
# we were expecting an error, confirm the error message
check_error_msg( expected_error )
return dlg # nb: the dialog is left on-screen
elif expected_constraints:
# we were expecting constraint warnings, confirm them
check_constraint_warnings( "Do you want to update this article?",
expected_constraints, "ok" if accept_constraints else "cancel"
)
return dlg # nb: the dialog is left on-screen
else:
# we were expecting the update to work, confirm this
expected = "updated OK" if sr else "created OK"
wait_for( 2,
lambda: check_toast( toast_type, expected, contains=True )
)
wait_for_not_elem( 2, "#article-form" )
return None
def _update_values( dlg, vals ):
"""Update an article's values in the form."""
for key,val in vals.items():
if key == "image":
if val:
change_image( dlg, val )
else:
remove_image( dlg )
elif key in ("publication", "publisher"):
row = find_child( ".row.{}".format( key ), dlg )
select = ReactSelect( find_child( ".react-select", row ) )
if not select.select.is_displayed():
key2 = "publisher" if key == "publication" else "publication"
row2 = find_child( ".row.{}".format( key2 ), dlg )
find_child( "label.parent-mode", row2 ).click()
wait_for( 2, select.select.is_displayed )
select.select_by_name( val )
elif key in ["authors","scenarios","tags"]:
select = ReactSelect( find_child( ".row.{} .react-select".format(key), dlg ) )
select.update_multiselect_values( *val )
else:
if key == "snippet":
sel = ".row.snippet textarea"
elif key == "pageno":
sel = "input.pageno"
else:
sel = ".row.{} input".format( key )
set_elem_text( find_child( sel, dlg ), val )
# ---------------------------------------------------------------------
def _check_sr( sr, expected ): #pylint: disable=too-many-return-statements
"""Check a search result."""
# check the title and subtitle
names = get_search_result_names( [sr] )
if names[0] != expected[0]:
return False
elem = find_child( ".subtitle", sr )
if expected[1]:
if not elem or elem.text != expected[1]:
return False
else:
if elem is not None:
return False
# check the snippet
if find_child( ".snippet", sr ).text != expected[2]:
return False
# check the authors
authors = [ t.text for t in find_children( ".author", sr ) ]
if authors != expected[4]:
return False
# check the tags
tags = [ t.text for t in find_children( ".tag", sr ) ]
if tags != expected[5]:
return False
# check the article's link
elem = find_child( "a.open-link", sr )
if expected[6]:
assert elem
if elem.get_attribute( "href" ) != expected[6]:
return False
else:
assert elem is None
return True