How to use SessionMap class of org.openqa.selenium.grid.sessionmap package

Best Selenium code snippet using org.openqa.selenium.grid.sessionmap.SessionMap

copy

Full Screen

...34import org.openqa.selenium.grid.server.EventBusConfig;35import org.openqa.selenium.grid.server.EventBusFlags;36import org.openqa.selenium.grid.server.HelpFlags;37import org.openqa.selenium.grid.server.Server;38import org.openqa.selenium.grid.sessionmap.SessionMap;39import org.openqa.selenium.grid.sessionmap.config.SessionMapFlags;40import org.openqa.selenium.grid.sessionmap.config.SessionMapOptions;41import org.openqa.selenium.remote.http.HttpClient;42import org.openqa.selenium.remote.tracing.DistributedTracer;43import org.openqa.selenium.remote.tracing.GlobalDistributedTracer;44@AutoService(CliCommand.class)45public class DistributorServer implements CliCommand {46 @Override47 public String getName() {48 return "distributor";49 }50 @Override51 public String getDescription() {52 return "Adds this server as the distributor in a selenium grid.";53 }54 @Override55 public Executable configure(String... args) {56 HelpFlags help = new HelpFlags();57 BaseServerFlags serverFlags = new BaseServerFlags(5553);58 SessionMapFlags sessionMapFlags = new SessionMapFlags();59 EventBusFlags eventBusFlags = new EventBusFlags();60 JCommander commander = JCommander.newBuilder()61 .programName(getName())62 .addObject(help)63 .addObject(eventBusFlags)64 .addObject(sessionMapFlags)65 .addObject(serverFlags)66 .build();67 return () -> {68 try {69 commander.parse(args);70 } catch (ParameterException e) {71 System.err.println(e.getMessage());72 commander.usage();73 return;74 }75 if (help.displayHelp(commander, System.out)) {76 return;77 }78 Config config = new CompoundConfig(79 new EnvConfig(),80 new ConcatenatingConfig("distributor", '.', System.getProperties()),81 new AnnotatedConfig(help),82 new AnnotatedConfig(eventBusFlags),83 new AnnotatedConfig(serverFlags),84 new AnnotatedConfig(sessionMapFlags),85 new DefaultDistributorConfig());86 LoggingOptions loggingOptions = new LoggingOptions(config);87 loggingOptions.configureLogging();88 DistributedTracer tracer = loggingOptions.getTracer();89 GlobalDistributedTracer.setInstance(tracer);90 EventBusConfig events = new EventBusConfig(config);91 EventBus bus = events.getEventBus();92 HttpClient.Factory clientFactory = HttpClient.Factory.createDefault();93 SessionMap sessions = new SessionMapOptions(config).getSessionMap(clientFactory);94 Distributor distributor = new LocalDistributor(95 tracer,96 bus,97 clientFactory,98 sessions);99 BaseServerOptions serverOptions = new BaseServerOptions(config);100 Server<?> server = new BaseServer<>(serverOptions);101 server.setHandler(distributor);102 server.start();103 };104 }105}...

Full Screen

Full Screen
copy

Full Screen

...33import org.openqa.selenium.grid.server.BaseServerFlags;34import org.openqa.selenium.grid.server.BaseServerOptions;35import org.openqa.selenium.grid.server.HelpFlags;36import org.openqa.selenium.grid.server.Server;37import org.openqa.selenium.grid.sessionmap.SessionMap;38import org.openqa.selenium.grid.sessionmap.config.SessionMapFlags;39import org.openqa.selenium.grid.sessionmap.config.SessionMapOptions;40import org.openqa.selenium.remote.http.HttpClient;41import org.openqa.selenium.remote.tracing.DistributedTracer;42import org.openqa.selenium.remote.tracing.GlobalDistributedTracer;43@AutoService(CliCommand.class)44public class RouterServer implements CliCommand {45 @Override46 public String getName() {47 return "router";48 }49 @Override50 public String getDescription() {51 return "Creates a router to front the selenium grid.";52 }53 @Override54 public Executable configure(String... args) {55 HelpFlags help = new HelpFlags();56 BaseServerFlags serverFlags = new BaseServerFlags(4444);57 SessionMapFlags sessionMapFlags = new SessionMapFlags();58 DistributorFlags distributorFlags = new DistributorFlags();59 JCommander commander = JCommander.newBuilder()60 .programName(getName())61 .addObject(help)62 .addObject(serverFlags)63 .addObject(sessionMapFlags)64 .addObject(distributorFlags)65 .build();66 return () -> {67 try {68 commander.parse(args);69 } catch (ParameterException e) {70 System.err.println(e.getMessage());71 commander.usage();72 return;73 }74 if (help.displayHelp(commander, System.out)) {75 return;76 }77 Config config = new CompoundConfig(78 new EnvConfig(),79 new ConcatenatingConfig("router", '.', System.getProperties()),80 new AnnotatedConfig(help),81 new AnnotatedConfig(serverFlags),82 new AnnotatedConfig(sessionMapFlags),83 new AnnotatedConfig(distributorFlags));84 LoggingOptions loggingOptions = new LoggingOptions(config);85 loggingOptions.configureLogging();86 DistributedTracer tracer = loggingOptions.getTracer();87 GlobalDistributedTracer.setInstance(tracer);88 HttpClient.Factory clientFactory = HttpClient.Factory.createDefault();89 SessionMapOptions sessionsOptions = new SessionMapOptions(config);90 SessionMap sessions = sessionsOptions.getSessionMap(clientFactory);91 BaseServerOptions serverOptions = new BaseServerOptions(config);92 DistributorOptions distributorOptions = new DistributorOptions(config);93 Distributor distributor = distributorOptions.getDistributor(tracer, clientFactory);94 Router router = new Router(tracer, clientFactory, sessions, distributor);95 Server<?> server = new BaseServer<>(serverOptions);96 server.setHandler(router);97 server.start();98 };99 }100}...

Full Screen

Full Screen
copy

Full Screen

...26import org.openqa.selenium.events.EventBus;27import org.openqa.selenium.events.local.GuavaEventBus;28import org.openqa.selenium.grid.data.Session;29import org.openqa.selenium.grid.data.SessionClosedEvent;30import org.openqa.selenium.grid.sessionmap.local.LocalSessionMap;31import org.openqa.selenium.grid.sessionmap.remote.RemoteSessionMap;32import org.openqa.selenium.grid.testing.PassthroughHttpClient;33import org.openqa.selenium.remote.SessionId;34import org.openqa.selenium.remote.http.HttpClient;35import org.openqa.selenium.remote.tracing.DistributedTracer;36import org.openqa.selenium.support.ui.FluentWait;37import org.openqa.selenium.support.ui.Wait;38import java.net.URI;39import java.net.URISyntaxException;40import java.util.UUID;41/​**42 * We test the session map by ensuring that the HTTP protocol is properly adhered to. If this is43 * true, then any implementations are interoperable, and we can breathe a sigh of relief.44 */​45public class SessionMapTest {46 private SessionId id;47 private Session expected;48 private SessionMap local;49 private HttpClient client;50 private SessionMap remote;51 private EventBus bus;52 @Before53 public void setUp() throws URISyntaxException {54 id = new SessionId(UUID.randomUUID());55 expected = new Session(56 id,57 new URI("http:/​/​localhost:1234"),58 new ImmutableCapabilities());59 bus = new GuavaEventBus();60 local = new LocalSessionMap(61 DistributedTracer.builder().build(),62 bus);63 client = new PassthroughHttpClient(local);64 remote = new RemoteSessionMap(client);65 }66 @Test67 public void shouldBeAbleToAddASession() {68 assertTrue(remote.add(expected));69 assertEquals(expected, local.get(id));70 }71 @Test72 public void shouldBeAbleToRetrieveASessionUri() {73 local.add(expected);74 assertEquals(expected, remote.get(id));75 }76 @Test77 public void shouldThrowANoSuchSessionExceptionIfSessionCannotBeFound() {78 assertThatExceptionOfType(NoSuchSessionException.class).isThrownBy(() -> local.get(id));79 assertThatExceptionOfType(NoSuchSessionException.class).isThrownBy(() -> remote.get(id));80 }81 @Test82 public void shouldAllowSessionsToBeRemoved() {83 local.add(expected);84 assertEquals(expected, remote.get(id));85 remote.remove(id);86 assertThatExceptionOfType(NoSuchSessionException.class).isThrownBy(() -> local.get(id));87 assertThatExceptionOfType(NoSuchSessionException.class).isThrownBy(() -> remote.get(id));88 }89 /​**90 * This is because multiple areas within the grid may all try and remove a session.91 */​92 @Test93 public void removingASessionThatDoesNotExistIsNotAnError() {94 remote.remove(id);95 }96 @Test(expected = NoSuchSessionException.class)97 public void shouldThrowAnExceptionIfGettingASessionThatDoesNotExist() {98 remote.get(id);99 }100 @Test101 public void shouldAllowEntriesToBeRemovedByAMessage() {102 local.add(expected);103 bus.fire(new SessionClosedEvent(expected.getId()));104 Wait<SessionMap> wait = new FluentWait<>(local).withTimeout(ofSeconds(2));105 wait.until(sessions -> {106 try {107 sessions.get(expected.getId());108 return false;109 } catch (NoSuchSessionException e) {110 return true;111 }112 });113 }114}...

Full Screen

Full Screen
copy

Full Screen

...19import org.openqa.selenium.grid.config.Config;20import org.openqa.selenium.grid.data.Session;21import org.openqa.selenium.grid.log.LoggingOptions;22import org.openqa.selenium.grid.server.NetworkOptions;23import org.openqa.selenium.grid.sessionmap.SessionMap;24import org.openqa.selenium.grid.sessionmap.config.SessionMapOptions;25import org.openqa.selenium.grid.web.Values;26import org.openqa.selenium.internal.Require;27import org.openqa.selenium.remote.SessionId;28import org.openqa.selenium.remote.http.HttpClient;29import org.openqa.selenium.remote.http.HttpRequest;30import org.openqa.selenium.remote.http.HttpResponse;31import org.openqa.selenium.remote.tracing.HttpTracing;32import org.openqa.selenium.remote.tracing.Tracer;33import java.io.UncheckedIOException;34import java.lang.reflect.Type;35import java.net.MalformedURLException;36import java.net.URI;37import static org.openqa.selenium.remote.http.Contents.asJson;38import static org.openqa.selenium.remote.http.HttpMethod.DELETE;39import static org.openqa.selenium.remote.http.HttpMethod.GET;40import static org.openqa.selenium.remote.http.HttpMethod.POST;41public class RemoteSessionMap extends SessionMap {42 private final HttpClient client;43 public RemoteSessionMap(Tracer tracer, HttpClient client) {44 super(tracer);45 this.client = Require.nonNull("HTTP client", client);46 }47 public static SessionMap create(Config config) {48 Tracer tracer = new LoggingOptions(config).getTracer();49 URI uri = new SessionMapOptions(config).getSessionMapUri();50 HttpClient.Factory clientFactory = new NetworkOptions(config).getHttpClientFactory(tracer);51 try {52 return new RemoteSessionMap(tracer, clientFactory.createClient(uri.toURL()));53 } catch (MalformedURLException e) {54 throw new UncheckedIOException(e);55 }56 }57 @Override58 public boolean isReady() {59 try {60 return client.execute(new HttpRequest(GET, "/​readyz")).isSuccessful();61 } catch (RuntimeException e) {62 return false;63 }64 }65 @Override66 public boolean add(Session session) {...

Full Screen

Full Screen
copy

Full Screen

...32import org.openqa.selenium.grid.server.EventBusConfig;33import org.openqa.selenium.grid.server.EventBusFlags;34import org.openqa.selenium.grid.server.HelpFlags;35import org.openqa.selenium.grid.server.Server;36import org.openqa.selenium.grid.sessionmap.SessionMap;37import org.openqa.selenium.grid.sessionmap.local.LocalSessionMap;38import org.openqa.selenium.remote.tracing.DistributedTracer;39import org.openqa.selenium.remote.tracing.GlobalDistributedTracer;40@AutoService(CliCommand.class)41public class SessionMapServer implements CliCommand {42 @Override43 public String getName() {44 return "sessions";45 }46 @Override47 public String getDescription() {48 return "Adds this server as the session map in a selenium grid.";49 }50 @Override51 public Executable configure(String... args) {52 HelpFlags help = new HelpFlags();53 BaseServerFlags serverFlags = new BaseServerFlags(5556);54 EventBusFlags eventBusFlags = new EventBusFlags();55 JCommander commander = JCommander.newBuilder()56 .programName(getName())57 .addObject(help)58 .addObject(serverFlags)59 .addObject(eventBusFlags)60 .build();61 return () -> {62 try {63 commander.parse(args);64 } catch (ParameterException e) {65 System.err.println(e.getMessage());66 commander.usage();67 return;68 }69 if (help.displayHelp(commander, System.out)) {70 return;71 }72 Config config = new CompoundConfig(73 new EnvConfig(),74 new ConcatenatingConfig("sessions", '.', System.getProperties()),75 new AnnotatedConfig(help),76 new AnnotatedConfig(serverFlags),77 new AnnotatedConfig(eventBusFlags),78 new DefaultSessionMapConfig());79 LoggingOptions loggingOptions = new LoggingOptions(config);80 loggingOptions.configureLogging();81 DistributedTracer tracer = loggingOptions.getTracer();82 GlobalDistributedTracer.setInstance(tracer);83 EventBusConfig events = new EventBusConfig(config);84 EventBus bus = events.getEventBus();85 SessionMap sessions = new LocalSessionMap(tracer, bus);86 BaseServerOptions serverOptions = new BaseServerOptions(config);87 Server<?> server = new BaseServer<>(serverOptions);88 server.setHandler(sessions);89 server.start();90 };91 }92}...

Full Screen

Full Screen
copy

Full Screen

...16/​/​ under the License.17package org.openqa.selenium.grid.sessionmap.config;18import org.openqa.selenium.grid.config.Config;19import org.openqa.selenium.grid.config.ConfigException;20import org.openqa.selenium.grid.sessionmap.SessionMap;21import org.openqa.selenium.grid.sessionmap.remote.RemoteSessionMap;22import org.openqa.selenium.remote.http.HttpClient;23import java.net.MalformedURLException;24import java.net.URL;25import java.util.Optional;26public class SessionMapOptions {27 private final Config config;28 public SessionMapOptions(Config config) {29 this.config = config;30 }31 public SessionMap getSessionMap(HttpClient.Factory clientFactory) {32 HttpClient client = clientFactory.createClient(getSessionMapUrl());33 return new RemoteSessionMap(client);34 }35 private URL getSessionMapUrl() {36 Optional<URL> host = config.get("sessions", "host").map(str -> {37 try {38 return new URL(str);39 } catch (MalformedURLException e) {40 throw new ConfigException("Session map server URI is not a valid URI: " + str);41 }42 });43 if (host.isPresent()) {44 return host.get();45 }46 Optional<Integer> port = config.getInt("sessions", "port");47 Optional<String> hostname = config.get("sessions", "hostname");48 if (!(port.isPresent() && hostname.isPresent())) {49 throw new ConfigException("Unable to determine host and port for the session map server");...

Full Screen

Full Screen

SessionMap

Using AI Code Generation

copy

Full Screen

1import java.util.Objects;2import java.util.Optional;3import java.util.Set;4import java.util.UUID;5import java.util.function.Consumer;6import org.openqa.selenium.NoSuchSessionException;7import org.openqa.selenium.grid.data.Session;8import org.openqa.selenium.grid.data.SessionId;9import org.openqa.selenium.grid.session.SessionRequest;10import org.openqa.selenium.grid.session.SessionRequestHandler;11import org.openqa.selenium.grid.session.SessionRequestHandlerFactory;12import org.openqa.selenium.grid.session.SessionRequestHandlerOptions;13import org.openqa.selenium.internal.Require;14import org.openqa.selenium.remote.SessionId;15import org.openqa.selenium.remote.http.HttpRequest;16import org.openqa.selenium.remote.http.HttpResponse;17import com.google.common.collect.ImmutableSet;18public class SessionMap implements SessionRequestHandlerFactory {19 private final SessionMapOptions options;20 public SessionMap(SessionMapOptions options) {21 this.options = Require.nonNull("Options", options);22 }23 public SessionRequestHandler apply(SessionRequestHandlerOptions handlerOptions) {24 return new SessionMapHandler(options, handlerOptions);25 }26 private static class SessionMapHandler implements SessionRequestHandler {27 private final SessionMapOptions options;28 private final SessionRequestHandlerOptions handlerOptions;29 private SessionMapHandler(SessionMapOptions options, SessionRequestHandlerOptions handlerOptions) {30 this.options = Require.nonNull("Options", options);31 this.handlerOptions = Require.nonNull("Handler options", handlerOptions);32 }33 public Optional<HttpResponse> execute(HttpRequest req, SessionRequest sessionRequest) {34 Require.nonNull("HTTP request", req);35 Require.nonNull("Session request", sessionRequest);36 SessionId sessionId = sessionRequest.getSessionId();37 if (sessionId == null) {38 sessionId = new SessionId(UUID.randomUUID());

Full Screen

Full Screen

SessionMap

Using AI Code Generation

copy

Full Screen

1package org.openqa.selenium.grid.sessionmap;2import java.net.URI;3import java.util.Map;4import java.util.Objects;5import java.util.Optional;6import java.util.concurrent.ConcurrentHashMap;7import org.openqa.selenium.grid.data.Session;8import org.openqa.selenium.grid.data.SessionId;9import org.openqa.selenium.internal.Require;10import org.openqa.selenium.remote.http.HttpClient;11import org.openqa.selenium.remote.tracing.Tracer;12public class SessionMap implements org.openqa.selenium.grid.sessionmap.SessionMap {13 private final Map<SessionId, Session> sessions = new ConcurrentHashMap<>();14 private final HttpClient.Factory clientFactory;15 private final Tracer tracer;16 private final URI uri;17 public SessionMap(HttpClient.Factory clientFactory, Tracer tracer, URI uri) {18 this.clientFactory = Require.nonNull("HTTP client factory", clientFactory);19 this.tracer = Require.nonNull("Tracer", tracer);20 this.uri = Require.nonNull("URI", uri);21 }22 public Session add(Session session) {23 Require.nonNull("Session", session);24 SessionId id = session.getId();25 Session existing = sessions.putIfAbsent(id, session);26 if (existing != null) {27 throw new SessionNotCreatedException("Session " + id + " already exists");28 }29 return session;30 }31 public Optional<Session> get(SessionId id) {32 Require.nonNull("Session ID", id);33 return Optional.ofNullable(sessions.get(id));34 }35 public Session remove(SessionId id) {36 Require.nonNull("Session ID", id);37 Session session = sessions.remove(id);38 if (session == null) {39 throw new NoSuchSessionException("Session " + id + " does not exist");40 }41 return session;42 }43 public Session update(Session session) {44 Require.nonNull("Session", session);45 SessionId id = session.getId();46 if (!sessions.containsKey(id)) {47 throw new NoSuchSessionException("Session " + id + " does not exist");48 }49 sessions.put(id, session);50 return session;51 }52 public boolean equals(Object o) {53 if (!(o instanceof SessionMap)) {54 return false;55 }56 SessionMap that = (SessionMap) o;57 return Objects.equals(this.sessions, that.sessions);58 }59 public int hashCode() {

Full Screen

Full Screen
copy
1 when(LoggerFactory.getLogger(GoodbyeController.class)).thenReturn(loggerMock);2
Full Screen

StackOverFlow community discussions

Questions
Discussion

Explicit Wait for findElements in Selenium Webdriver

How to check if an element is visible with WebDriver

Selenium + JUnit: test order/flow?

Selenium/WebView get clicked element

How do I prevent Selenium RC from stealing window focus while my tests are running?

Selenium webdriver: finding all elements with similar id

How to Add Selenium Library in Java or Android Project. The import org.openqa cannot be resolved

Selenium RC - Error: Could not find or load main class jar

How to get element&#39;s text without retrieving the text from its descendant

Unable to perform any actions on elements which were located under flexbox

You can do something like this:

//explicit wait for input field field
WebDriverWait wait = new WebDriverWait(driver, 10);
wait.until(ExpectedConditions.visibilityOfElementLocated(By.tagName("input")));

ExpectedConditions class can be useful in a lot of cases and provides some set of predefined condition to wait for the element. Here are some of them:

  • alertIsPresent : Alert is present
  • elementSelectionStateToBe: an element state is selection.
  • elementToBeClickable: an element is present and clickable.
  • elementToBeSelected: element is selected
  • frameToBeAvailableAndSwitchToIt: frame is available and frame selected.
  • invisibilityOfElementLocated: an element is invisible
  • presenceOfAllElementsLocatedBy: present element located by.
  • textToBePresentInElement: text present on particular an element
  • textToBePresentInElementValue: and element value present for a particular element.
  • visibilityOf: an element visible.
  • titleContains: title contains
https://stackoverflow.com/questions/34397608/explicit-wait-for-findelements-in-selenium-webdriver

Blogs

Check out the latest blogs from LambdaTest on this topic:

Which Browsers Are Important For Your Cross Browser Testing?

This article is a part of our Content Hub. For more in-depth resources, check out our content hub on Cross Browser Testing Tutorial.

Easily Execute Python UnitTest Parallel Testing In Selenium

This article is a part of our Content Hub. For more in-depth resources, check out our content hub on Selenium Python Tutorial.

What is a WebView And How To Test It?

Convenience is something that we can never be fully satisfied with. This is why software developers are always made to push their limits for bringing a better user experience, without compromising the functionality. All for the sake of saving the churn in today’s competitive business. People are greedy for convenience and this is why Hybrid applications have been so congenial in the cyber world.

16 Best Practices Of CI/CD Pipeline To Speed Test Automation

Every software project involves some kind of ‘processes’ & ‘practices’ for successful execution & deployment of the project. As the size & scale of the project increases, the degree of complications also increases in an exponential manner. The leadership team should make every possible effort to develop, test, and release the software in a manner so that the release is done in an incremental manner thereby having minimal (or no) impact on the software already available with the customer.

Automated Cross Browser Testing

Testing a website in a single browser using automation script is clean and simple way to accelerate your testing. With a single click you can test your website for all possible errors without manually clicking and navigating to web pages. A modern marvel of software ingenuity that saves hours of manual time and accelerate productivity. However for all this magic to happen, you would need to build your automation script first.

Selenium 4 Tutorial:

LambdaTest’s Selenium 4 tutorial is covering every aspects of Selenium 4 testing with examples and best practices. Here you will learn basics, such as how to upgrade from Selenium 3 to Selenium 4, to some advanced concepts, such as Relative locators and Selenium Grid 4 for Distributed testing. Also will learn new features of Selenium 4, such as capturing screenshots of specific elements, opening a new tab or window on the browser, and new protocol adoptions.

Chapters:

  1. Upgrading From Selenium 3 To Selenium 4?: In this chapter, learn in detail how to update Selenium 3 to Selenium 4 for Java binding. Also, learn how to upgrade while using different build tools such as Maven or Gradle and get comprehensive guidance for upgrading Selenium.

  2. What’s New In Selenium 4 & What’s Being Deprecated? : Get all information about new implementations in Selenium 4, such as W3S protocol adaption, Optimized Selenium Grid, and Enhanced Selenium IDE. Also, learn what is deprecated for Selenium 4, such as DesiredCapabilites and FindsBy methods, etc.

  3. Selenium 4 With Python: Selenium supports all major languages, such as Python, C#, Ruby, and JavaScript. In this chapter, learn how to install Selenium 4 for Python and the features of Python in Selenium 4, such as Relative locators, Browser manipulation, and Chrom DevTool protocol.

  4. Selenium 4 Is Now W3C Compliant: JSON Wireframe protocol is retiring from Selenium 4, and they are adopting W3C protocol to learn in detail about the advantages and impact of these changes.

  5. How To Use Selenium 4 Relative Locator? : Selenium 4 came with new features such as Relative Locators that allow constructing locators with reference and easily located constructors nearby. Get to know its different use cases with examples.

  6. Selenium Grid 4 Tutorial For Distributed Testing: Selenium Grid 4 allows you to perform tests over different browsers, OS, and device combinations. It also enables parallel execution browser testing, reads up on various features of Selenium Grid 4 and how to download it, and runs a test on Selenium Grid 4 with best practices.

  7. Selenium Video Tutorials: Binge on video tutorials on Selenium by industry experts to get step-by-step direction from automating basic to complex test scenarios with Selenium.

Selenium 101 certifications:

LambdaTest also provides certification for Selenium testing to accelerate your career in Selenium automation testing.

Run Selenium automation tests on LambdaTest cloud grid

Perform automation testing on 3000+ real desktop and mobile devices online.

...Most popular Stackoverflow questions on SessionMap

Most used methods in SessionMap

Test Your Web Or Mobile Apps On 3000+ Browsers

Signup for free

Try LambdaTest Now !!

Get 100 minutes of automation test minutes FREE!!

Next-Gen App & Browser Testing Cloud

Was this article helpful?

Helpful

NotHelpful