- Added a new
Stop
Action, which can take a Question and Resolution and will stop the test until that assertion is true (or a timeout is reached). Or justStop()
and it will wait until you press enter.
- The typehint for the
Matches
Resolution now includes re.compile objects. ..._to_log
methods are now all property-ized, which fixes a few logging errors.
- Pruned the Sphinx directives because they don't need to be as long as i thought they did.
- Slimmed down the documentation by noting method aliases in the method's docstring and excluding the aliases from the documentation (since thye're copies of the original method).
MakeNote
now correctly logs what key the Actor is jotting something down under.
- We now make use of the
Self
type fromtyping_extensions
, to better type-hint subclassed entites in ScreenPy.
- Did quite a lot of project standardization (big big thanks @bandophahita!).
- Updated all the dependencies to be more up-to-date!
- Specifically updated Pydantic to use the new Pydantic-Settings module for settings management. Waaay slicker.
- A new
Either
Action! For those times where you might want to do one thing or a different thing depending on the current state of the application. Or any other way to use it you can think of!
- Almost all the improvements this time were clerical: improved the docs setup, added linting to the tests, add
ruff
...q - Added some logic to better represent the values from the test in the Narrator's logs (thanks @bandophahita!).
- Added support for Python 3.12!
- Fixed several typing issues.
- Fixed an issue where
Silently
wasn't correctly wrapping Describables, so the wrong message was being read by the Narrator (thanks @bandophahita!). - Fixed some cases where
beat
should be loggingNone
but wasn't, and then fixed a bug with that where it was loggingNone
too often iffrom __future__ import annotations
was at the top of the file (thanks again, @bandophahita!). - Fixed an issue where
Eventually
would log multiple of the same exceptions. We actually completely reworked the logic here to make things much easier to read (thanks again again, @bandophahita!).
ContainsTheEntry
was still inheriting fromBaseResolution
, even after the deprecation (thanks for the issue, @jardilac91!). No longer!
BaseResolution
is now deprecated in favor of the newResolvable
protocol. No longer are Resolutions the odd-duck of ScreenPy, now they behave the same as Actions (Performable
) and Questions (Answerable
)! See the deprecations page entry for more information.- As foretold, the
has_cleanup_tasks
method ofActor
has been removed. Er, what do you mean i forgot to mention it last year when it was deprecated?? Well hopefully you saw the warning! 😬
- Actors are now callable, and-
- There are now present-tense aliases for all Actions, so you can do
the_actor(Sees(This(), IsCool()))
! Well except thatThis
is not a Question andIsCool
is not a Resolution... unless you make them! - Added
Silently
"Action" (more of an Adverb, really). This wrapper silences all narration for the -ables contained within. No Actions, Questions, or Resolutions will be narrated inside ofSilently
! - Added
Log
Action to log the answer to a Question real quick. Could be useful during debugging.
- Almost everything you could want from ScreenPy is now available from the top-level. Now instead of doing
from screenpy.actions import ...
,from screenpy.resolutions import...
, you can just do it all usingfrom screenpy import ...
in one block! You can still do the old way if you want, but now you can do it this way, too! SeeAnyOf
andSeeAllOf
now behave consistently with each other and consistently with how one might expectand
andor
to behave. They both accept 0 tests (and pass), they both allow 1 test, and they both short-circuit their testing. They also handle narrating plurals better!- Settings now use Pydantic, and are much, MUCH better! You can set settings for ScreenPy through environment variables, through pyproject.toml, or in a conftest.py file. Checkout the updated Settings documentation for details!
Eventually
now preserves the order of the errors it encounters, which will make debugging much easier.Eventually
now correctly sets both poll rate and timeout in the same call, if you want to do that sort of thing.
- Gravitas can now be found in its own module,
narration.gravitas
, or imported directly fromscreenpy.narration (e.g.
from screenpy.narration import AIRY`). It was a little weird for them to live in the Narrator file.
- Turns out that the
aside
function didn't takegravitas
, even though the Adapter itself did. That's fixed, and also the Narrator and all the tests are updated to fix that as well, and the Adapter Protocol definition, too.
- Er, the convenient imports from the previous build caused some circular imports.
screenpy.actions
andscreenpy.resolutions
still need you to import them through the full path. - Python <3.10 don't like the
|
typehint union; switch back toUnion
for those.
- Dropping support for Python 3.7. ScreenPy will probably still work in Python 3.7, but we're no longer making sure it does.
- Added a lot of new Resolutions for text and number comparisons:
- StartsWith
- EndsWith
- Matches for you regex-speakers
- ContainsItemMatching for strings-in-lists regex assertions
- IsLessThan and IsLessThanOrEqualTo
- IsGreaterThan and IsGreaterThanOrEqualTo
- IsInRange
- Added convenience imports, so you can import everything directly from screenpy (from screenpy import Actor, given, See...)
StdOutAdapter
(now also calledLoggingAdapter
for clarity) can now acceptgravitas
,AIRY
being equivalent tologging.DEBUG
.- All
staticmethods
on Actor, Action, and Resolution classes (maybe more? All of them!) are nowclassmethods
and appropriately typed, which should make it much easier (and possible at all) to subclass, if you need to. Big big thanks to @bandophahita for that huge effort!
- Added an
ErrorKeeper
protocol, currently only for Questions that encounter an error while being answered. This is now used byMakeNote
to mention the exception, which will help debug some confusing test states. (h/t @bandophahita!) - Added official "extras" support for
screenpy_playwright
, install withpip install screenpy[playwright]
!
- A significant improvement on test coverage via unit tests! (h/t @bandophahita!)
- Added new aliases for
IsCloseTo
andHasLength
to use in their negated cases:CloseTo
andHaveLength
. - Added official "extras" support for
screenpy_appium
, install withpip install screenpy[appium]
!
- Tons of typing improvements thanks to @bandophahita, especially with Resolutions! Things should be a lot nicer in your IDEs.
- Added a count to
Eventually
that mentions how many timesEventually
looped before failing. This should be helpful in the case where the performable that should eventually complete takes longer than the timeout set forEventually
.
2day is 2sday, by the way. Also I pushed this up at 22:22!
- ScreenPy is now broken up in2 its core (this repository),
screenpy_selenium
,screenpy_requests
,screenpy_pyotp
, andscreenpy_adapter_allure
. This will cause some big, big breakages in your suites, and i'm very sorry about it. To help get you back up and running, see the changes at https://screenpy-docs.readthedocs.io/en/latest/deprecations.html#breaking-changes - Those deprecated methods mentioned on that same deprecations page are also finally removed.
- Yet another rewrite of the documentation! This time it's much more focused, since the documentation is only for the base module.
- The rewrite includes a new Complete Example section, which shows a fictional example of a ScreenPy test file and breaks down each part of it.
- Eventually now also tells you how many times it attempted its performable, which could be useful for debugging in certain situations.
- If
See
was given a value to compare against a Resolution, the value is now properly printed to the log instead of just the string "{value}". 🤦
- Added
.of_the_first_of_the
aliases to several Actions and Questions, to allow for clearer readings of tests. - Added an
aside
of the actual value whenSee
is about to compare a pre-baked value instead of asking a Question. - Fixed
IsNot
's descriptions, especially when they were describing when the negative (... of the negative).
- Fixes the Narrator skipping narrations if they occurred within a double-kinked cable. In fact, completely rewrote how the Narrator handles multiple cable kinks in a much more understandable way (h/t @bandophahita).
- Another bugfix for the Narrator when reporting errors -- if the user isn't using Allure, an error in the script would cause a
StopIteration
error to be raised aftewards. Pretty annoying when debugging! - Speaking of errors being raised, if an error were raised while the Narrator's mic cable was kinked or while the Narrator was off the air, the Narrator would never narrate again (until the next script ran anyway). That's fixed!
- Fixed up the quick hackjob for narrating errors to put it in a much more elegant spot.
- Kind of critical bugfix for the Narrator's new ability to report errors -- the Narrator now only reports errors if their mic cable is not kinked and they are on air. This was causing some pretty unfortunate interactions with
Eventually
...
- The
clean_up
method that all Actors use to clean up after themselves was mutating the cleanup task list as it was iterating through it. This was not good! Actors have been reminded of proper looping etiquette.
Eventually
now raises aDeliveryError
to bring it in line withWait
and all other Actions which raise errors.Eventually
now mentions all of the exceptions it encountered when it raisesDeliveryError
.- Several Actions have less-awkwardly-worded
beat
s!
MakeNote
andSee
both relied on catching anAttributeError
to figure out if they had been passed a value or a Question. But if the Question itself encountered anAttributeError
while being answered, a very confusing situation would arise that was very annoying to debug. Now these two Actions usehasattr
instead!- The
AllureAdapter
is no longer falsely marking steps which failed as having passed! Bringin' back them reds and yellows.
- The
Adapter
protocol now supports anerror
method, which allows adapters to report exceptions in their own way.
- Toned down the aggressive pruning by
get_additive_description
. It was cutting off closing quotes! - Fixed the incorrect method being called when attaching in the
AllureAdapter
.
- Added some nice new Actions for reporting:
AttachTheFile
,SaveScreenshot
, andSaveConsoleLog
!
Eventually
andSee
now have beats! They try their best to make a natural-sounding English statement about what your Actor is doing, but if you need to tweak the message...- Performables and Answerables may now have a
describe
function, whichEventually
andSee
both directly use. - Narrators can now
attach
files! Calling Narrator'sattach
method will broadcast the filepath to all the adapters plugged into their microphone. Adapters can look for specific keywords, if needed—all other adapters will ignore those keywords.
- I was not able to figure out how to fix the problem with the Allure reporting, #74. :( (Edit: This is Perry from the Future! I figured it out!)
- Updated
See
andMakeNote
typehints now that they allow for bare values and Questions.
- The gravitas levels have moved from
screenpy.pacing
toscreenpy.narration.narrator
, and they've also changed terminology. The updated terminology is more generic, describing how serious the gravitas is. The gravitas level is passed along to the adapters to be used or ignored as needed.
- Added a
Narrator
class, who handles all the reporting! The Narrator's microphone can be outfitted with different adapters to report to whatever reporting service you want. Included are the newAllureAdapter
andStdOutAdapter
, which will maintain existing behavior. - (h/t @langgaibo) Added an
IsCloseTo
resolution! - (h/t @bandophahita) Added an
Eventually
action!
- Part of the above
Narrator
addition is that there is now a way for you to manually control logging, either by putting the narratoroff_the_air
or by making theirmicrophone_cable_kinked
. It was immediately handy forEventually
, maybe you'll find it useful, too! - Better typehints for
Target
! - Better
beat
forMakeNote
! See
andMakeNote
can now both accept values and Questions!
- XPATH locators which begin with a parenthesis are now correctly recognized as XPATH locators.
- I'm on a tear, eh?
- Added a more useful test in
screenpy-quickstart
that shows off more actions. - Adjusted the return type of the
noted_under
direction fromobject
toAny
. I think it could be something more specific, butAny
is the best i've got for now.
- Added
.of_all
method to the Attribute Question because that is, incidentally, the way i needed to be able to use it. Woops!
- (h/t @langgaibo and @sramdas-dod) Actors can now clean up after themselves! Give 'em a task with
.has_cleanup_task(CleanUpTask())
and they'll do theCleanUpTask()
task as they exit. Or sooner, if you callActor.cleans_up()
yourself! (CleanUpTask
not included.) - (h/t @langgaibo) Actors can now take notes and use them... anywhere. The new MakeNote Action allows you to save the answer to a Question under a handy key, which can then be recalled later using the
noted_under
direction! Which brings me to... - The Director! The Director is omnipresent and singular. They are always watching, waiting to take or give a needed note. Ask for one by using a
the_noted(key)
ornoted_under(key)
direction, wherever you need the information they hold. 🔺👁🔺 - Added the See, SeeAllOf, and SeeAnyOf Actions, to relieve Actors of the burden of asserting. Take a look at the new Deprecations docs page to learn more.
- Added the Attribute Question, to ask about an element's attributes. Like "value", which is the one i'll be asking about.
- (h/t @WandyYing and @bandophahita) SOLIDified Actors a little bit more by deprecating their asserting responsibility. This was part of adding the new See etc. Actions mentioned above. Actors now have a
should
method, which is really just an alias forwas_able_to
andattempts_to
, but is meant to be used with that See Action. Or not, it's really up to you. - Targets can now be used as if they themselves were locator tuples. You can do
target[0]
and pass in*target
s. It's wild! And cleans up a really gross list comprehension from the Wait source code. - As ever, another huge documentation update. Things should be more consistent now. Maybe. (I'll probably find more things to smooth out as soon as i commit this.) (Edit: i did.)
- Deprecated
Actor.should_see_the
andActor.should_see_any_of
in favor of usingActor.should
with the new See, SeeAllOf, and SeeAnyOf Actions. These methods will be removed in 4.0.0. Instructions for modifying your tests can be found in the 3.1.0 Deprecations documentation
- Missed a
from typing import Protocol
, that's fixed now. Woops!
- (h/t @bandophahita) CLI logging now has indentation levels to help show which statements are encapsulated by Questions, tasks, Actions, etc.!
- Reverted the
Protocol
import fromtyping
back totyping_extensions
for better compatibility. This will be revisited when Python 4 is released, probably.
- (h/t @bandophahita) New IsClickable Resolution to check if an element is clickable!
- Wait now only puts one "..." at the end of its messages. :eye_roll:
mypy
will now be looking at theexamples/
directory as well, so it can catch errors like...
- (h/t @bandophahita) Removed the BaseAction, BaseAbility, and BaseQuestion classes, for sure this time.
mypy
should no longer complain whenever you use any of the SendMETHODRequest API request Actions (e.g. SendGETRequest, SendPOSTRequest, etc.).
- Wait can do custom log messages again, to make the reports look nicer.
- Element now catches the correct exception type, now that Target's the one doing the finding.
- Wait now formats its custom exception correctly.
- What's a release without some same-day bug fixes, right?
- BrowseTheWeb no longer has
.to_find()
or.to_wait_for()
. I realized these methods actually break the S in SOLID, and these methods were limiting Actions from really doing what they're supposed to do. See the Improvements section below for the direct benefit of this change!
- Wait can now use strategies that don't use a Target, or have multiple arguments. You can now use any of Selenium's URL-monitoring expected conditions or any other custom strategies through
Wait().using(strategy).with_("any", "number", "of", "args")
! - AddHeader can now accept new forms of header-setting arguments, like a dict or just alternating pairs. Now
AddHeader(Cookie="yes=please")
==AddHeader({"Cookie": "yes=please"})
==AddHeader("Cookie", "yes=please")
! - Added SetHeaders Action, if you want to set the headers exactly (and remove all unmentioned headers).
- AddHeader, SetHeaders, and Send[METHOD]Request Actions now all have a way to mark their contents as secret, so they won't be logged. Add a
.which_should_be_kept_secret()
or.secretly()
to the end and the log won't mention the unmentionables. - I did another docs overhaul, this time making them read in a way that might please Hemingway.
- Click is now able to be chained without a Target to click. This was preventing clicking at an offset, which annoyed me at least.
- Updated all the dependencies to support Python 3.9! Kind of a boring update.
- Made the reported Actions more consistently worded.
- Removed
on_top
method from SwitchToTab, it wasn't consistent under certain circumstances and that could be misleading. - Removed the BaseAction, BaseAbility, and BaseQuestion classes in favor of Protocols.
- Added API testing support!
- Added MakeAPIRequests Ability.
- Added SendGETRequest, SendPOSTRequest, SendPATCHRequest, SendPUTRequest, SendOPTIONSRequest, SendHEADRequest, SendDELETERequest, and SendAPIRequest Actions.
- Added AddHeader/AddHeaders Action.
- Added Cookies, CookiesOnTheWebSession, CookiesOnTheAPISession Questions.
- Added StatusCodeOfTheLastResponse Question.
- Added BodyOfTheLastResponse Question.
- Added HeadersOfTheLastResponse Question.
- Added ContainsTheKey, ContainsTheValue, ContainsTheEntry Resolutions.
- Added
should_see_any_of
assertion method to Actors, which is similar toshould_see_the
but passes if at least one of its tests are true, instead of all of them. - Added HasLength Resolution, for fun.
- Added a "cookbook" section to the docs to give examples of common use-cases.
- Huge docs overhaul, aimed at reducing word count and increasing word value.
- Switched to using Protocols for type hinting instead of base classes.
- IsVisible Resolution now has a nicer mismatch message.
- Fixed a timing issue with has_method_with_return_value where the return value could change by the time the mismatch message was being written to the log, which made it look like a passing test was failing.
- Fixed
SwitchTo.default()
logging a very metaphysical "{The Actor} switches to the None". - Fixed Enter's logging when
then_hits
is used to hit a key without a nice text representation (like "Return"). - Fixed a small collection of copy/paste errors in documentation and logged strings.
- Added try/except to the Element Question so it will return
None
if the element doesn't exist, which enablesIsNot(Visible())
orIsNot(EqualTo(None))
to test for non-visibility/presence of an element. - Moved BrowsingError to
screenpy.exceptions
too, missed that one!
- Removed
.then_wait_for
methods from Enter and Click. - Removed all
to_switch_to...
methods from BrowseTheWeb, preferring to use the browser directly. Also removedto_visit
andto_get
from BrowseTheWeb for the same reason. - Changed UnableToPerformError and UnableToActError to UnableToPerform and UnableToAct.
- Moved DeliveryError and TargetingError to
screenpy.exceptions
(though you can still technically import them fromscreenpy.actor
andscreenpy.target
). - Removed several syntactic sugar options that, upon retrospect, aren't really grammatically possible in this module.
- Added so many new Actions!
- Added RefreshPage Action.
- Added GoBack Action.
- Added GoForward Action.
- Added SwitchToTab Action. (It turns out SwitchTo didn't quite switch to all the things you needed to switch to.)
- Added Chain Action, to do chained Actions! Such as these:
- Added DoubleClick Action.
- Added RightClick Action.
- Added MoveMouse/Hover Action.
- Added HoldDown Action.
- Added Release Action.
- Enabled several existing Actions to be chained with the new Chain Action:
- Click
- Enter
- Enter2FAToken
- Pause
- Added Element Question, to test things about a specific element, such as...
- Added IsVisible Resolution, to test if an element is visible!
- You can now see the same detailed logs without the Allure report if you wish! All the Action is now logged at INFO level. Try adding
--log-cli-level=INFO
to yourpytest
run, for example!
- Cleaned up a lot of documentation.
- Updated the
screenpy-quickstart
script to better reflect the current state and recommendations.
- Fixed a strange issue that prevented Allure from properly naming the acts and scenes in the Behavior view.
- "Properly" typed ScreenPy. It could probably be done a little better, though.
- Added IOS and Android browser support! 🎉 (Will add full Appium support later!)
- Added Pause Action, for those times you really need it.
- Added Debug Action, to drop a debugger in the middle of a chain of Actions/tasks.
- Added SwitchTo Action, to switch to iframes, back to the default frame, and whatever else you need to switch to.
- Added AcceptAlert/DismissAlert/RespondToPrompt to support any javascript alert Actions you could want.
- Added a bunch of new syntactic sugar and default options.
- Added a new method to Target that will allow you to pass in your own locator tuple, like
Target.located((By.LINK_TEXT, "click me"))
- Added additional logging to improve debugging experience.
- Added custom exceptions with exception chaining to provide more context when tests break.
- HUGE documentation improvement.
- Typed ScreenPy, and marked ScreenPy as being typed!
- Fixed strange grammar in logged Resolution lines.
- Deprecated the
then_wait_for
methods on Click and Enter now that we have a Wait Action. These methods will be removed in 2.0.0. Instructions for modifying your tests can be found in the 1.0.0 Deprecations documentation
- Added the Wait Action, which enables the Actor to wait for a Target to be visible, invisible, or contain expected text.
- Added the Clear Action, which enables the Actor to clear text from an input field.
- Added import magic to make importing Actions, Resolutions, Targets, ... everything! easier.
- Added some natural-language-enabling syntactic sugar to Resolutions.
- Added some extra flavor to Actors being created because i can't help myself.
- Added "the-internet" example
- Added Selected Question
- Adjusted strategy for multi-Target Text Question
- Added more consistency in coding style
- Fixed issues with Select Actions
- Fixed a lot of issues with documentation. I think i understand reStructured Text now. Mostly.
- Fixed call to old keyword argument. Woops.
- Added more Resolutions.
- Added List Question.
- Added more syntactic sugar.
- May have gone overboard with additional theming.
- Added Sphinx documentation!
- Added docstrings and type hints across the board.
- Reorganized method lists to be more consistent.
- Fixed missing requirements installation in setup.py
- Upload to personal git repository
- Publish to PyPI
- Too little sleep
- Conception
- Demo to coworkers