All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.openqa.selenium.grid.node.Node Maven / Gradle / Ivy

Go to download

Selenium automates browsers. That's it! What you do with that power is entirely up to you.

The newest version!
// Licensed to the Software Freedom Conservancy (SFC) under one
// or more contributor license agreements.  See the NOTICE file
// distributed with this work for additional information
// regarding copyright ownership.  The SFC licenses this file
// to you under the Apache License, Version 2.0 (the
// "License"); you may not use this file except in compliance
// with the License.  You may obtain a copy of the License at
//
//   http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing,
// software distributed under the License is distributed on an
// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
// KIND, either express or implied.  See the License for the
// specific language governing permissions and limitations
// under the License.

package org.openqa.selenium.grid.node;

import static org.openqa.selenium.remote.HttpSessionId.getSessionId;
import static org.openqa.selenium.remote.http.Contents.asJson;
import static org.openqa.selenium.remote.http.Route.combine;
import static org.openqa.selenium.remote.http.Route.delete;
import static org.openqa.selenium.remote.http.Route.get;
import static org.openqa.selenium.remote.http.Route.matching;
import static org.openqa.selenium.remote.http.Route.post;

import com.google.common.collect.ImmutableMap;
import java.io.IOException;
import java.net.URI;
import java.time.Duration;
import java.util.Map;
import java.util.ServiceLoader;
import java.util.Set;
import java.util.UUID;
import java.util.logging.Logger;
import java.util.stream.Collectors;
import java.util.stream.StreamSupport;
import org.openqa.selenium.BuildInfo;
import org.openqa.selenium.Capabilities;
import org.openqa.selenium.NoSuchSessionException;
import org.openqa.selenium.WebDriverException;
import org.openqa.selenium.grid.data.CreateSessionRequest;
import org.openqa.selenium.grid.data.CreateSessionResponse;
import org.openqa.selenium.grid.data.NodeId;
import org.openqa.selenium.grid.data.NodeStatus;
import org.openqa.selenium.grid.data.Session;
import org.openqa.selenium.grid.security.RequiresSecretFilter;
import org.openqa.selenium.grid.security.Secret;
import org.openqa.selenium.internal.Either;
import org.openqa.selenium.internal.Require;
import org.openqa.selenium.io.TemporaryFilesystem;
import org.openqa.selenium.json.Json;
import org.openqa.selenium.remote.SessionId;
import org.openqa.selenium.remote.http.HttpRequest;
import org.openqa.selenium.remote.http.HttpResponse;
import org.openqa.selenium.remote.http.Routable;
import org.openqa.selenium.remote.http.Route;
import org.openqa.selenium.remote.locators.CustomLocator;
import org.openqa.selenium.remote.tracing.SpanDecorator;
import org.openqa.selenium.remote.tracing.Tracer;
import org.openqa.selenium.status.HasReadyState;

/**
 * A place where individual webdriver sessions are running. Those sessions may be in-memory, or only
 * reachable via localhost and a network. Or they could be something else entirely.
 *
 * 

This class responds to the following URLs: * *

* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
VerbURL TemplateMeaning
POST/se/grid/node/sessionAttempts to start a new session for the given node. The posted data should be a * json-serialized {@link Capabilities} instance. Returns a serialized {@link Session}. * Subclasses of {@code Node} are expected to register the session with the * {@link org.openqa.selenium.grid.sessionmap.SessionMap}.
GET/se/grid/node/session/{sessionId}Finds the {@link Session} identified by {@code sessionId} and returns the JSON-serialized * form.
DELETE/se/grid/node/session/{sessionId}Stops the {@link Session} identified by {@code sessionId}. It is expected that this will * also cause the session to removed from the * {@link org.openqa.selenium.grid.sessionmap.SessionMap}.
GET/se/grid/node/owner/{sessionId}Allows the node to be queried about whether or not it owns the {@link Session} identified * by {@code sessionId}. This returns a boolean.
POST/se/grid/node/connection/{sessionId}Allows the node to be ask about whether or not new websocket connections are allowed for the {@link Session} * identified by {@code sessionId}. This returns a boolean.
*/session/{sessionId}/*The request is forwarded to the {@link Session} identified by {@code sessionId}. When the * Quit command is called, the {@link Session} should remove itself from the * {@link org.openqa.selenium.grid.sessionmap.SessionMap}.
*/ public abstract class Node implements HasReadyState, Routable { private static final Logger LOG = Logger.getLogger(Node.class.getName()); private static final BuildInfo INFO = new BuildInfo(); private static final ImmutableMap OS_INFO = loadOsInfo(); protected final Tracer tracer; private final NodeId id; private final URI uri; private final Duration sessionTimeout; private final Route routes; protected boolean draining; protected Node( Tracer tracer, NodeId id, URI uri, Secret registrationSecret, Duration sessionTimeout) { this.tracer = Require.nonNull("Tracer", tracer); this.id = Require.nonNull("Node id", id); this.uri = Require.nonNull("URI", uri); this.sessionTimeout = Require.positive("Session timeout", sessionTimeout); Require.nonNull("Registration secret", registrationSecret); RequiresSecretFilter requiresSecret = new RequiresSecretFilter(registrationSecret); Set customLocators = StreamSupport.stream(ServiceLoader.load(CustomLocator.class).spliterator(), false) .collect(Collectors.toSet()); if (!customLocators.isEmpty()) { String names = customLocators.stream() .map(CustomLocator::getLocatorName) .collect(Collectors.joining(", ")); LOG.info("Binding additional locator mechanisms: " + names); } Json json = new Json(); routes = combine( // "getSessionId" is aggressive about finding session ids, so this needs to be the last // route that is checked. matching( req -> getSessionId(req.getUri()) .map(SessionId::new) .map(sessionId -> this.getSession(sessionId) != null) .orElse(false)) .to(() -> new ForwardWebDriverCommand(this)) .with(spanDecorator("node.forward_command")), new CustomLocatorHandler(this, registrationSecret, customLocators), post("/session/{sessionId}/se/file") .to(params -> new UploadFile(this, sessionIdFrom(params))) .with(spanDecorator("node.upload_file")), get("/session/{sessionId}/se/files") .to(params -> new DownloadFile(this, sessionIdFrom(params))) .with(spanDecorator("node.download_file")), post("/session/{sessionId}/se/files") .to(params -> new DownloadFile(this, sessionIdFrom(params))) .with(spanDecorator("node.download_file")), delete("/session/{sessionId}/se/files") .to(params -> new DownloadFile(this, sessionIdFrom(params))) .with(spanDecorator("node.download_file")), get("/se/grid/node/owner/{sessionId}") .to(params -> new IsSessionOwner(this, sessionIdFrom(params))) .with(spanDecorator("node.is_session_owner").andThen(requiresSecret)), post("/se/grid/node/connection/{sessionId}") .to(params -> new TryAcquireConnection(this, sessionIdFrom(params))) .with(spanDecorator("node.is_session_owner").andThen(requiresSecret)), delete("/se/grid/node/session/{sessionId}") .to(params -> new StopNodeSession(this, sessionIdFrom(params))) .with(spanDecorator("node.stop_session").andThen(requiresSecret)), get("/se/grid/node/session/{sessionId}") .to(params -> new GetNodeSession(this, sessionIdFrom(params))) .with(spanDecorator("node.get_session").andThen(requiresSecret)), post("/se/grid/node/session") .to(() -> new NewNodeSession(this, json)) .with(spanDecorator("node.new_session").andThen(requiresSecret)), post("/se/grid/node/drain") .to(() -> new Drain(this, json)) .with(spanDecorator("node.drain").andThen(requiresSecret)), get("/se/grid/node/status") .to(() -> req -> new HttpResponse().setContent(asJson(getStatus()))) .with(spanDecorator("node.node_status")), get("/status").to(() -> new StatusHandler(this)).with(spanDecorator("node.status"))); } private static ImmutableMap loadOsInfo() { return ImmutableMap.of( "arch", System.getProperty("os.arch"), "name", System.getProperty("os.name"), "version", System.getProperty("os.version")); } private SessionId sessionIdFrom(Map params) { return new SessionId(params.get("sessionId")); } private SpanDecorator spanDecorator(String name) { return new SpanDecorator(tracer, req -> name); } public NodeId getId() { return id; } public URI getUri() { return uri; } public String getNodeVersion() { return String.format("%s (revision %s)", INFO.getReleaseLabel(), INFO.getBuildRevision()); } public ImmutableMap getOsInfo() { return OS_INFO; } public abstract Either newSession( CreateSessionRequest sessionRequest); public abstract HttpResponse executeWebDriverCommand(HttpRequest req); public abstract Session getSession(SessionId id) throws NoSuchSessionException; public TemporaryFilesystem getUploadsFilesystem(SessionId id) throws IOException { throw new UnsupportedOperationException(); } public TemporaryFilesystem getDownloadsFilesystem(UUID uuid) throws IOException { throw new UnsupportedOperationException(); } public abstract HttpResponse uploadFile(HttpRequest req, SessionId id); public abstract HttpResponse downloadFile(HttpRequest req, SessionId id); public abstract void stop(SessionId id) throws NoSuchSessionException; public abstract boolean isSessionOwner(SessionId id); public abstract boolean tryAcquireConnection(SessionId id); public abstract boolean isSupporting(Capabilities capabilities); public abstract NodeStatus getStatus(); public abstract HealthCheck getHealthCheck(); public Duration getSessionTimeout() { return sessionTimeout; } public boolean isDraining() { return draining; } public abstract void drain(); @Override public boolean matches(HttpRequest req) { return routes.matches(req); } @Override public HttpResponse execute(HttpRequest req) { return routes.execute(req); } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy