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

org.openqa.grid.internal.BaseRemoteProxy Maven / Gradle / Ivy

Go to download

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

There is a newer version: 4.0.0-alpha-2
Show 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.grid.internal;

import static org.openqa.grid.common.RegistrationRequest.MAX_INSTANCES;

import com.google.gson.JsonObject;
import com.google.gson.JsonParser;

import org.apache.http.HttpEntity;
import org.apache.http.HttpHost;
import org.apache.http.HttpResponse;
import org.apache.http.client.HttpClient;
import org.apache.http.message.BasicHttpRequest;
import org.apache.http.util.EntityUtils;
import org.openqa.grid.common.RegistrationRequest;
import org.openqa.grid.common.SeleniumProtocol;
import org.openqa.grid.common.exception.GridException;
import org.openqa.grid.internal.listeners.TimeoutListener;
import org.openqa.grid.internal.utils.CapabilityMatcher;
import org.openqa.grid.internal.utils.DefaultHtmlRenderer;
import org.openqa.grid.internal.utils.HtmlRenderer;
import org.openqa.grid.internal.utils.configuration.GridNodeConfiguration;
import org.openqa.selenium.remote.DesiredCapabilities;
import org.openqa.selenium.remote.internal.HttpClientFactory;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.net.MalformedURLException;
import java.net.URL;
import java.security.InvalidParameterException;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.logging.Level;
import java.util.logging.Logger;

public class BaseRemoteProxy implements RemoteProxy {
  private final RegistrationRequest request;

  private static final Logger log = Logger.getLogger(BaseRemoteProxy.class.getName());

  // the host the remote listen on.The final URL will be proxy.host + slot.path
  protected volatile URL remoteHost;

  protected final GridNodeConfiguration config;

  // list of the type of test the remote can run.
  private final List testSlots;

  private final Registry registry;


  private final String id;

  private volatile boolean stop = false;
  private CleanUpThread cleanUpThread;

  public List getTestSlots() {
    return testSlots;
  }

  public Registry getRegistry() {
    return registry;
  }

  public CapabilityMatcher getCapabilityHelper() {
    return registry.getConfiguration().capabilityMatcher;
  }


  /**
   * Create the proxy from the info sent by the remote. 

If maxSession is not specified, default * to 1 = max number of tests running at a given time will be 1.

For each capability, * maxInstances is defaulted to 1 if not specified = max number of test of each capability running * at a time will be 1. maxInstances for firefox can be > 1. IE won't support it. * * @param request The request * @param registry The registry to use */ public BaseRemoteProxy(RegistrationRequest request, Registry registry) { this.request = request; this.registry = registry; this.config = new GridNodeConfiguration(); // the registry is the 'hub' configuration, which is used as a seed. this.config.merge(registry.getConfiguration()); // the proxy values must override any that the hub specify where an overlap occurs. // merging last causes the values to be overridden. this.config.merge(request.getConfiguration()); // host and port are merge() protected values -- overrule this behavior this.config.host = request.getConfiguration().host; this.config.port = request.getConfiguration().port; String url = config.getRemoteHost(); String id = config.id; if (url == null && id == null) { throw new GridException( "The registration request needs to specify either the remote host, or a valid id."); } if (url != null) { try { this.remoteHost = new URL(url); } catch (MalformedURLException e) { // should only happen when a bad config is sent. throw new GridException("Not a correct url to register a remote : " + url); } } // if id was provided in the request, use that if (id != null) { this.id = id; } else { // otherwise assign the remote host as id. this.id = remoteHost.toExternalForm(); } List capabilities = request.getConfiguration().capabilities; List slots = new ArrayList<>(); for (DesiredCapabilities capability : capabilities) { Object maxInstance = capability.getCapability(MAX_INSTANCES); SeleniumProtocol protocol = SeleniumProtocol.fromCapabilitiesMap(capability.asMap()); if (maxInstance == null) { log.warning("Max instance not specified. Using default = 1 instance"); maxInstance = "1"; } int value = Integer.parseInt(maxInstance.toString()); for (int i = 0; i < value; i++) { Map c = new HashMap<>(); for (String k : capability.asMap().keySet()) { c.put(k, capability.getCapability(k)); } slots.add(createTestSlot(protocol, c)); } } this.testSlots = Collections.unmodifiableList(slots); } public void setupTimeoutListener() { cleanUpThread = null; if (this instanceof TimeoutListener) { if (config.cleanUpCycle > 0 && config.timeout > 0) { log.fine("starting cleanup thread"); cleanUpThread = new CleanUpThread(this); new Thread(cleanUpThread, "RemoteProxy CleanUpThread for " + getId()) .start(); // Thread safety reviewed (hopefully ;) } } } public String getId() { if (id == null) { throw new RuntimeException("Bug. Trying to use the id on a proxy but it hasn't been set."); } return id; } public void teardown() { stop = true; } /** * Internal use only */ public void forceSlotCleanerRun() { cleanUpThread.cleanUpAllSlots(); } class CleanUpThread implements Runnable { private BaseRemoteProxy proxy; public CleanUpThread(BaseRemoteProxy proxy) { this.proxy = proxy; } public void run() { log.fine("cleanup thread starting..."); while (!proxy.stop) { try { Thread.sleep(config.cleanUpCycle); } catch (InterruptedException e) { log.severe("clean up thread died. " + e.getMessage()); } cleanUpAllSlots(); } } void cleanUpAllSlots() { for (TestSlot slot : getTestSlots()) { try { cleanUpSlot(slot); } catch (Throwable t) { log.warning("Error executing the timeout when cleaning up slot " + slot + t.getMessage()); } } } private void cleanUpSlot(TestSlot slot) { TestSession session = slot.getSession(); if (session != null) { long inactivity = session.getInactivityTime(); boolean hasTimedOut = inactivity > getTimeOut(); if (hasTimedOut) { if (!session.isForwardingRequest()) { log.logp(Level.WARNING, "SessionCleanup", null, "session " + session + " has TIMED OUT due to client inactivity and will be released."); try { ((TimeoutListener) proxy).beforeRelease(session); } catch(IllegalStateException ignore) { log.log(Level.WARNING, ignore.getMessage()); } registry.terminate(session, SessionTerminationReason.TIMEOUT); } } if (session.isOrphaned()) { log.logp(Level.WARNING, "SessionCleanup", null, "session " + session + " has been ORPHANED and will be released"); try { ((TimeoutListener) proxy).beforeRelease(session); } catch(IllegalStateException ignore) { log.log(Level.WARNING, ignore.getMessage()); } registry.terminate(session, SessionTerminationReason.ORPHAN); } } } } public GridNodeConfiguration getConfig() { return config; } public RegistrationRequest getOriginalRegistrationRequest() { return request; } public int getMaxNumberOfConcurrentTestSessions() { return config.maxSession; } public URL getRemoteHost() { return remoteHost; } public TestSession getNewSession(Map requestedCapability) { log.fine("Trying to create a new session on node " + this); if (!hasCapability(requestedCapability)) { log.fine("Node " + this + " has no matching capability"); return null; } // any slot left at all? if (getTotalUsed() >= config.maxSession) { log.fine("Node " + this + " has no free slots"); return null; } // any slot left for the given app ? for (TestSlot testslot : getTestSlots()) { TestSession session = testslot.getNewSession(requestedCapability); if (session != null) { return session; } } return null; } public int getTotalUsed() { int totalUsed = 0; for (TestSlot slot : getTestSlots()) { if (slot.getSession() != null) { totalUsed++; } } return totalUsed; } public boolean hasCapability(Map requestedCapability) { for (TestSlot slot : getTestSlots()) { if (slot.matches(requestedCapability)) { return true; } } return false; } public boolean isBusy() { return getTotalUsed() != 0; } /** * Takes a registration request and return the RemoteProxy associated to it. It can be any class * extending RemoteProxy. * * @param request The request * @param registry The registry to use * @param RemoteProxy subclass * @return a new instance built from the request. */ @SuppressWarnings("unchecked") public static T getNewInstance( RegistrationRequest request, Registry registry) { try { String proxyClass = request.getConfiguration().proxy; if (proxyClass == null) { log.fine("No proxy class. Using default"); proxyClass = BaseRemoteProxy.class.getCanonicalName(); } Class clazz = Class.forName(proxyClass); log.fine("Using class " + clazz.getName()); Object[] args = new Object[]{request, registry}; Class[] argsClass = new Class[]{RegistrationRequest.class, Registry.class}; Constructor c = clazz.getConstructor(argsClass); Object proxy = c.newInstance(args); if (proxy instanceof RemoteProxy) { ((RemoteProxy) proxy).setupTimeoutListener(); return (T) proxy; } throw new InvalidParameterException("Error: " + proxy.getClass() + " isn't a remote proxy"); } catch (InvocationTargetException e) { throw new InvalidParameterException("Error: " + e.getTargetException().getMessage()); } catch (Exception e) { throw new InvalidParameterException("Error: " + e.getMessage()); } } @Override public int hashCode() { final int prime = 31; int result = 1; result = prime * result + ((id == null) ? 0 : id.hashCode()); return result; } @Override public boolean equals(Object obj) { if (this == obj) { return true; } if (obj == null) { return false; } if (getClass() != obj.getClass()) { return false; } RemoteProxy other = (RemoteProxy) obj; if (getId() == null) { if (other.getId() != null) { return false; } } else if (!getId().equals(other.getId())) { return false; } return true; } // less busy to more busy. public int compareTo(RemoteProxy o) { if (o == null) { return -1; } return (int)(getResourceUsageInPercent() - o.getResourceUsageInPercent()); } @Override public String toString() { return getRemoteHost() != null ? getRemoteHost().toString() : ""; } private final HtmlRenderer renderer = new DefaultHtmlRenderer(this); public HtmlRenderer getHtmlRender() { return renderer; } public int getTimeOut() { return config.timeout * 1000; } public HttpClientFactory getHttpClientFactory() { return getRegistry().getHttpClientFactory(); } /** * @throws GridException If the node if down or doesn't recognize the /wd/hub/status request. */ public JsonObject getStatus() throws GridException { String url = getRemoteHost().toExternalForm() + "/wd/hub/status"; BasicHttpRequest r = new BasicHttpRequest("GET", url); HttpClient client = getHttpClientFactory().getGridHttpClient(config.nodeStatusCheckTimeout, config.nodeStatusCheckTimeout); HttpHost host = new HttpHost(getRemoteHost().getHost(), getRemoteHost().getPort(), getRemoteHost().getProtocol()); HttpResponse response; String existingName = Thread.currentThread().getName(); HttpEntity entity = null; try { Thread.currentThread().setName("Probing status of " + url); response = client.execute(host, r); entity = response.getEntity(); int code = response.getStatusLine().getStatusCode(); if (code == 200) { JsonObject status = new JsonObject(); try { status = extractObject(response); } catch (Exception e) { // ignored due it's not required from node to return anything. Just 200 code is enough. } EntityUtils.consume(response.getEntity()); return status; } else if (code == 404) { // selenium RC case JsonObject status = new JsonObject(); EntityUtils.consume(response.getEntity()); return status; } else { EntityUtils.consume(response.getEntity()); throw new GridException("server response code : " + code); } } catch (Exception e) { throw new GridException(e.getMessage(), e); } finally { Thread.currentThread().setName(existingName); try { //Added by jojo to release connection thoroughly EntityUtils.consume(entity); } catch (IOException e) { log.info("Exception thrown when consume entity"); } } } private JsonObject extractObject(HttpResponse resp) throws IOException { BufferedReader rd = new BufferedReader(new InputStreamReader(resp.getEntity().getContent())); StringBuilder s = new StringBuilder(); String line; while ((line = rd.readLine()) != null) { s.append(line); } rd.close(); return new JsonParser().parse(s.toString()).getAsJsonObject(); } public float getResourceUsageInPercent() { return 100 * (float)getTotalUsed() / (float)getMaxNumberOfConcurrentTestSessions(); } public long getLastSessionStart() { long last = -1; for (TestSlot slot : getTestSlots()) { last = Math.max(last, slot.getLastSessionStart()); } return last; } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy