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

Add setOrientation command #2121

Open
wants to merge 6 commits into
base: main
Choose a base branch
from

Conversation

johntmcintosh
Copy link

@johntmcintosh johntmcintosh commented Nov 2, 2024

Proposed changes

Introduce a setOrientation command, accepting orientation values of portrait, upsideDown, landscapeLeft, landscapeRight.

If we're onboard with the syntax of this command and argument names, we should also make an addition to the public documentation repo to merge when this is ready to ship.

Background

After starting to build some flows thanks to the recent addition of landscape support on iOS, I found that it worked locally if I manually rotated the simulator into landscape orientation, but was unable to run on CI due to the simulator always launching in portrait.

#1221 started some discussion about adding the capability to force rotation, but at the time was blocked pending the baseline landscape support that was recently added.

Alternatives

Without adding this capability to maestro, I think the only way to run flows in landscape orientation on CI (at least on iOS simulators) is to expect users to rotate the simulator after it launches before starting the flows. Rotating the simulator is not available through xcrun simctl, which leaves AppleScript as the alternative. AppleScript requires accessibility permissions be granted to the CI runner. While an AppleScript approach should be feasible, it add complexity to all users CI setups by requiring accessibility permissions to run.

Testing

Unit tests: added new unit tests matching conventions from similar commands

iOS: verified locally using my own flows that without this command landscape tests fail, but with the addition of a call to setOrientation: landscapeLeft after launching the app, the tests succeed. The simulator frame stays in portrait orientation, but the test runner is now able to interact with the content correctly. This is the same behavior I see when running XCUITests on the landscape orientation app directly from Xcode.

Android: I do not have an android setup available and have NOT tested this implementation on Android. Android testing support requested!

CI

I get one failure from CI related not committing the compiled iOS changes. I tried including a commit of that, but got the same error, so I could use some guidance on how to proceed there (or if that needs to be re-compiled by a maintainer for consistent provisioning or similar).

Issues fixed

#1221

@johntmcintosh johntmcintosh force-pushed the feature/set-orientation branch from abcf6ea to 90b4677 Compare November 2, 2024 17:42
@vbarthel-fr
Copy link

vbarthel-fr commented Nov 6, 2024

👋 Hi there :) I'm just a random guy with no prior knowledge on maestro (I have been using maestro for a while though) and I'm also highly interested in having access to such setOrientation command. So thanks a lot for your contribution 👍 🔥

Android: I do not have an android setup available and have NOT tested this implementation on Android. Android testing support requested!

I do have access to an android setup, and I have made a first test: so far, so good!

Screenshot 2024-11-06 at 20 22 19 Screenshot 2024-11-06 at 20 22 38 Screenshot 2024-11-06 at 20 22 51 Screenshot 2024-11-06 at 20 23 06

(Tested from johntmcintosh:feature/set-orientation and running ./maestro studio)

Do not hesitate to tell me if I can be of any help 💪

[Edit 2024-11-07]: I have also tested on a real device (not an emulator) and it's also working like a charm.

@@ -601,6 +601,18 @@ class AndroidDriver(
}
}

override fun setOrientation(orientation: DeviceOrientation) {
// Disable accelerometer based rotation before overriding orientation
dadb.shell("settings put system accelerometer_rotation 0")
Copy link

@vbarthel-fr vbarthel-fr Nov 6, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Random thought - not sure it's something we should care about - but from my understanding the "auto-rotate" that is disabled here is never re-enabled.

That also seems to be true for the user_rotation system settings: the last setOrientation call will "persist" the chosen user_rotation system setting beyond the current running flow.

If someone is reusing the same android device to run multiple flows one after the other, it might introduce some side effect between them: if one flow is using setOrientation commands while another expect the orientation to be in "auto-rotate" mode (or expect a "default" orientation).

Edit: if this "side effect" is not something desired, it might be possible to mimic what is done for the setProxy: if set, proxy is "reset" when AndroidDriver is closed:


(setProxy is also using settings under the hood)

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hey @vbarthel-fr 👋 first off, thanks so much pulling and performing an android test for me!

You bring up a good topic here that I hadn't thought about yet. (And thanks for the pointer to resetProxy as an example being called from close().)

Do you foresee any holes in doing something like this:

  1. The first time that setOrientation is called for a flow we store the current values of accelerometer_rotation and user_rotation (and for iOS just the XCUIDevice.shared.orientation)
  2. From the close(), if those were saved we reset them to the original values.

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yep, that sounds good to me :)

But I'm not knowledgeable enough to know for sure that is OK to rely on AndroidDriver.close(). Tomorrow, I will try to have a closer look at the source code, starting by understanding when close() is called.

@vbarthel-fr
Copy link

I tried to implement an end-to-end test for Android : https://github.com/vbarthel-fr/maestro-e2e-set-orientation I tried my best to have the smallest APK as possible (8Kb in debug) that might even be added to this repo if it seems valuable.

The app observes the user_rotation system settings and display a human readable version of it in a text view, while the maestro script calls the setOrientation command and assert that the correct orientation is displayed.

 
 ║  > Flow: test-set-orientation-flow
 
 ║    ✅   Launch app "com.example.maestro.orientation"
 ║    ✅   Set orientation LANDSCAPE_LEFT
 ║    ✅   Assert that "LANDSCAPE_LEFT" is visible
 ║    ✅   Set orientation LANDSCAPE_RIGHT
 ║    ✅   Assert that "LANDSCAPE_RIGHT" is visible
 ║    ✅   Set orientation UPSIDE_DOWN
 ║    ✅   Assert that "UPSIDE_DOWN" is visible
 ║    ✅   Set orientation PORTRAIT
 ║    ✅   Assert that "PORTRAIT" is visible
 

hopefully it might be useful at some point :)

@Fishbowler Fishbowler added the v1.40.0 Release 1.40.0 label Dec 6, 2024
johntmcintosh and others added 5 commits December 31, 2024 16:50
I do not have an android setup and have not tested this yet. This implementation is based off conventions in `AndroidDriver.kt` for interacting with `adb` and this reference for what commands should be run https://stackoverflow.com/q/25864385.
@Fishbowler Fishbowler force-pushed the feature/set-orientation branch from 90b4677 to 2dff691 Compare December 31, 2024 17:05
@Fishbowler
Copy link
Contributor

Rebased

@Fishbowler
Copy link
Contributor

@johntmcintosh This is a terrific contribution, and high on the list of things folks have asked for! ❤️

@vbarthel-fr Awesome work on the testing! Do you mind if we fork the app and add the test into this PR as part of the e2e test run?

@johntmcintosh
Copy link
Author

@Fishbowler thanks! I have fairly limited availability for the next couple of weeks, but the outstanding topics that I'm aware of before being merge-ready were:

  1. This discussion about whether and how we should reset the Android configuration changes from close().
  2. My uncertainty with the correct way to commit the compiled iOS changes.
  3. Any additional e2e tests (like you referenced in your comment this morning).

Do you have availability to close out those 3 threads, or is my availability blocking for getting those done?

@Fishbowler
Copy link
Contributor

I've discussed this internally, and there's lots of things that can affect the end state of the test. Tracking them all automatically could be difficult, and in some cases, undesirable. Leaving it inside developer control seems good enough. I suspect that's what happens right now for airplane mode, for example.

Am more than happy to take on the other bits!

@vbarthel-fr
Copy link

@vbarthel-fr Awesome work on the testing! Do you mind if we fork the app and add the test into this PR as part of the e2e test run?

Thanks for the kind words :) And I don't mind if you fork the app and add the test into this PR! All the better if it can help!

@Fishbowler Fishbowler force-pushed the feature/set-orientation branch from 2e36ab2 to 28127d8 Compare January 6, 2025 09:59
@Fishbowler
Copy link
Contributor

Tests: done ✔️

We'll need a docs PR

@FatjonRrapaj
Copy link

Tests: done ✔️

We'll need a docs PR

Hello guys, thank you for this amazing contribution.

When will this be merged?

@Fishbowler
Copy link
Contributor

With the latest bugfix release out the door this morning, I'm switching focus back to features, and this one is high on my list! I'm hoping to have this merged before the end of the month - I want to check its compatibility with Robin first.

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

Successfully merging this pull request may close these issues.

4 participants