Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

'I press' step does not escape single quotes within text #137

Open
ThrawnCA opened this issue Jun 20, 2023 · 2 comments
Open

'I press' step does not escape single quotes within text #137

ThrawnCA opened this issue Jun 20, 2023 · 2 comments

Comments

@ThrawnCA
Copy link

There is logic in behaving/web/steps/links.py to handle single quotes (which will disrupt XPath) within the specified link text, but this logic is not present in the 'I press "{name"}' step in behaving/web/steps/forms.py, so usage like

I press "Dave's Books"

will fail with "Element not found".

ThrawnCA added a commit to qld-gov-au/ckanext-data-qld that referenced this issue Jun 20, 2023
- There is a bug in Behaving that breaks this step when single quotes are used.
Raised ggozad/behaving#137
@duttonw
Copy link
Collaborator

duttonw commented Jan 6, 2025

would you like to contribute a fix for this @ThrawnCA

@ThrawnCA
Copy link
Author

ThrawnCA commented Jan 6, 2025

@duttonw I don't have that access.

Suggested diff:

diff --git a/src/behaving/web/steps/forms.py b/src/behaving/web/steps/forms.py
index b31074f..aa37466 100644
--- a/src/behaving/web/steps/forms.py
+++ b/src/behaving/web/steps/forms.py
@@ -139,17 +139,22 @@ def i_focus(context, name):
 @when('I press "{name}"')
 @persona_vars
 def i_press(context, name):
+    # Use 'concat' to assemble search string if it contains single quotes.
+    # This will enable even a mixture of single and double quotes.
+    if "'" in name:
+        quoted_name = "concat('" + """', "'", '""".join(name.split("'")) + "')"
+    else:
+        quoted_name = f"'{name}'"
     element = context.browser.find_by_xpath(
         (
-            "//*[@id='%(name)s']|"
-            "//*[@name='%(name)s']|"
-            "//button[contains(string(), '%(name)s')]|"
-            "//input[@type='button' and contains(string(), '%(name)s')]|"
-            "//input[@type='button' and contains(@value, '%(name)s')]|"
-            "//input[@type='submit' and contains(@value, '%(name)s')]|"
-            "//a[contains(string(), '%(name)s')]"
+            f"//*[@id={quoted_name}]|"
+            f"//*[@name={quoted_name}]|"
+            f"//button[contains(string(), {quoted_name})]|"
+            f"//input[@type='button' and contains(string(), {quoted_name})]|"
+            f"//input[@type='button' and contains(@value, {quoted_name})]|"
+            f"//input[@type='submit' and contains(@value, {quoted_name})]|"
+            f"//a[contains(string(), {quoted_name})]"
         )
-        % {"name": name}
     )
     assert (
         element

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

2 participants