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

[WIP] core(fr): split out shared driver utilities #12366

Closed
wants to merge 1 commit into from

Conversation

patrickhulce
Copy link
Collaborator

@patrickhulce patrickhulce commented Apr 15, 2021

Summary
This is a draft preview of my rough plan for sharing the rest of driver utilities between legacy and FR. Essentially we're working toward a state where 100% of what's left in driver is specific to legacy operation and everything shared is extracted to rely on FRProtocolSession or ExecutionContext.

The feedback I am seeking at this stage is if this method extraction pattern feels reasonable to other driver API consumers as well (that's YOU 😄). The primary goal of this structure is to avoid a 2nd monolithic driver and provide a way of sharing complex protocol interaction in a composable way. I've converted a few of the usage examples so you can get a feel for how it changes driver usage.

Potential Future Work
Optional future work would be binding these modules of driver onto driver itself for example all the service worker methods might be available on driver.serviceWorkers.getRegistrations so that consumers don't have to pull in the module and mock every sendCommand. For rarely used methods I think this is unnecessary and leads to a monolithic API again, but we can handle that on an individual module basis.

Next Steps
I'll let this marinate for the week and start breaking things up on Monday if there are no objections.

@google-cla google-cla bot added the cla: yes label Apr 15, 2021
await driver.cacheNatives();
await driver.dismissJavaScriptDialogs();
await driver.registerRequestIdleCallbackWrap(options.settings);
await assertNoSameOriginServiceWorkerClients(driver.defaultSession, options.requestedUrl);
Copy link
Collaborator Author

Choose a reason for hiding this comment

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

this file is a good example of how things would change

Copy link
Member

@adamraine adamraine left a comment

Choose a reason for hiding this comment

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

Structure LGTM, some other comments I'll hold onto for the real PR.

@@ -0,0 +1,87 @@
/**
Copy link
Member

Choose a reason for hiding this comment

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

Some ideas:

  • page-content.js
  • content.js

@patrickhulce
Copy link
Collaborator Author

Thanks @adamraine ! sounds good I'll start breaking it up.

@brendankenny
Copy link
Member

The feedback I am seeking at this stage is if this method extraction pattern feels reasonable to other driver API consumers as well (that's YOU 😄). The primary goal of this structure is to avoid a 2nd monolithic driver and provide a way of sharing complex protocol interaction in a composable way. I've converted a few of the usage examples so you can get a feel for how it changes driver usage.

This move seems fine. Driver became kind of a utility dumping ground, but part of the reason the moves here are so clean is because a lot of the functions in there are already composable. To me the main issue with Driver (beyond just overdue general cleanups) is the complicated state that crept in with multi target and autoattach support (e.g. #7608 (review)). With that (hopefully) easier to follow and kept separate when built on top of sessions, the question here seems mostly one of should the utility functions be in one namespace or split up into multiple ones.

Multiple namespaces is usually(?) better, but OTOH when it's hard to come up with a name for a module containing getRequestContent() and scrollTo(), that means it'll also be hard to think of the name later when trying to find them and I'll probably just search for protocol names like I do today to see if we have any existing code using a particular command :) It's not perfect, but we also aren't making a puppeteer API in here for anyone, just collecting methods used in more than one place or enshrining things so the next person doesn't have to do the research, so either way seems fine, really.

I think the real ergonomic questions aren't going to be clear until things settle a little bit, so I'm very happy to defer on that. The things I'm really liking in here are getting rid of the cruft (I notice querySelector() and lh-element are finally getting axed :) and some of the clarifications (e.g. the much more clear emulation/throttling distinction).

  • I'd also love to see some of the gather-runner-specific logic pulled out (or at least considered for pulling out) of some of these and into the caller. e.g. should registerRequestIdleCallbackWrap really do nothing if settings.throttlingMethod !== 'simulate' or should gather-runner just not call it in that case?
  • People may have good arguments for not doing this (and it's probably a case-by-case call regardless), but at least my instinct with single-caller functions like getScrollPosition is to move them back to the caller rather than have them in a utility collection, and functions like enableRuntimeEvents don't seem like they even need a function for them at all.

@patrickhulce
Copy link
Collaborator Author

I'd also love to see some of the gather-runner-specific logic pulled out (or at least considered for pulling out) of some of these and into the caller. e.g. should registerRequestIdleCallbackWrap really do nothing if settings.throttlingMethod !== 'simulate' or should gather-runner just not call it in that case?

People may have good arguments for not doing this (and it's probably a case-by-case call regardless), but at least my instinct with single-caller functions like getScrollPosition is to move them back to the caller rather than have them in a utility collection, and functions like enableRuntimeEvents don't seem like they even need a function for them at all.

That was also on my target list next (sort of) :) My plan for gather-runner extraction was to eliminate these individual tiny ones and instead restructure them into the shared parts of gathering, but if we're already on the same page, then I'll probably just tackle those when I get to that splitting out for discussion.

@patrickhulce
Copy link
Collaborator Author

Everything substantive in this PR has landed 🎉

@patrickhulce patrickhulce deleted the fr_driver_refactor branch May 5, 2021 14:24
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants