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

org.opencastproject.serviceregistry.api.ServiceRegistryInMemoryImpl Maven / Gradle / Ivy

There is a newer version: 16.7
Show newest version
/*
 * Licensed to The Apereo Foundation under one or more contributor license
 * agreements. See the NOTICE file distributed with this work for additional
 * information regarding copyright ownership.
 *
 *
 * The Apereo Foundation licenses this file to you under the Educational
 * Community 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://opensource.org/licenses/ecl2.txt
 *
 * 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.opencastproject.serviceregistry.api;

import org.opencastproject.job.api.JaxbJob;
import org.opencastproject.job.api.Job;
import org.opencastproject.job.api.Job.Status;
import org.opencastproject.job.api.JobImpl;
import org.opencastproject.job.api.JobParser;
import org.opencastproject.job.api.JobProducer;
import org.opencastproject.security.api.Organization;
import org.opencastproject.security.api.OrganizationDirectoryService;
import org.opencastproject.security.api.SecurityService;
import org.opencastproject.security.api.User;
import org.opencastproject.security.api.UserDirectoryService;
import org.opencastproject.serviceregistry.api.SystemLoad.NodeLoad;
import org.opencastproject.util.NotFoundException;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.IOException;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.Date;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.LinkedHashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicLong;

/** Simple and in-memory implementation of a the service registry intended for testing scenarios. */
public class ServiceRegistryInMemoryImpl implements ServiceRegistry {

  /** Logging facility */
  private static final Logger logger = LoggerFactory.getLogger(ServiceRegistryInMemoryImpl.class);

  /** Default dispatcher timeout (1 second) */
  public static final long DEFAULT_DISPATCHER_TIMEOUT = 100;

  /** Hostname for localhost */
  private static final String LOCALHOST = "localhost";

  /** The hosts */
  protected Map hosts = new HashMap();

  /** The service registrations */
  protected Map> services = new HashMap>();

  /** The serialized jobs */
  protected Map jobs = new HashMap();

  /** A mapping of services to jobs */
  protected Map> jobHosts = new HashMap>();

  /** The thread pool to use for dispatching queued jobs. */
  protected ScheduledExecutorService dispatcher = Executors.newScheduledThreadPool(1);

  /** The job identifier */
  protected AtomicLong idCounter = new AtomicLong();

  /** Holds the current running job */
  protected Job currentJob = null;

  /**
   * An (optional) security service. If set to a non-null value, this will be used to obtain the current user when
   * creating new jobs.
   */
  protected SecurityService securityService = null;

  /** The user directory service */
  protected UserDirectoryService userDirectoryService = null;

  /** The organization directory service */
  protected OrganizationDirectoryService organizationDirectoryService = null;

  protected Incidents incidents;

  /**
   * A static list of statuses that influence how load balancing is calculated
   */
  protected static final List JOB_STATUSES_INFLUENCING_LOAD_BALANCING;

  static {
    JOB_STATUSES_INFLUENCING_LOAD_BALANCING = new ArrayList();
    JOB_STATUSES_INFLUENCING_LOAD_BALANCING.add(Status.QUEUED);
    JOB_STATUSES_INFLUENCING_LOAD_BALANCING.add(Status.RUNNING);
  }

  public ServiceRegistryInMemoryImpl(JobProducer service, float maxLoad, SecurityService securityService,
          UserDirectoryService userDirectoryService, OrganizationDirectoryService organizationDirectoryService,
          IncidentService incidentService) throws ServiceRegistryException {
    //Note: total memory here isn't really the correct value, but we just need something (preferably non-zero)
    registerHost(LOCALHOST, LOCALHOST, "Admin", Runtime.getRuntime().totalMemory(), Runtime.getRuntime().availableProcessors(), maxLoad);
    if (service != null)
      registerService(service, maxLoad);
    this.securityService = securityService;
    this.userDirectoryService = userDirectoryService;
    this.organizationDirectoryService = organizationDirectoryService;
    this.incidents = new Incidents(this, incidentService);
    this.dispatcher.scheduleWithFixedDelay(new JobDispatcher(), DEFAULT_DISPATCHER_TIMEOUT, DEFAULT_DISPATCHER_TIMEOUT,
            TimeUnit.MILLISECONDS);
  }

  public ServiceRegistryInMemoryImpl(JobProducer service, SecurityService securityService,
          UserDirectoryService userDirectoryService, OrganizationDirectoryService organizationDirectoryService,
          IncidentService incidentService)
          throws ServiceRegistryException {
    this(service, Runtime.getRuntime().availableProcessors(), securityService, userDirectoryService, organizationDirectoryService, incidentService);
  }

  /**
   * This method shuts down the service registry.
   */
  public void dispose() {
    if (dispatcher != null) {
      try {
        dispatcher.shutdownNow();
        if (!dispatcher.isShutdown()) {
          logger.info("Waiting for Dispatcher to terminate");
          dispatcher.awaitTermination(10, TimeUnit.SECONDS);
        }
      } catch (InterruptedException e) {
        logger.error("Error shutting down the Dispatcher", e);
      }
    }
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#enableHost(String)
   */
  @Override
  public void enableHost(String host) throws ServiceRegistryException, NotFoundException {
    if (hosts.containsKey(host)) {
      hosts.get(host).setActive(true);
    } else {
      throw new NotFoundException("The host named " + host + " was not found");
    }
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#disableHost(String)
   */
  @Override
  public void disableHost(String host) throws ServiceRegistryException, NotFoundException {
    if (hosts.containsKey(host)) {
      hosts.get(host).setActive(false);
    } else {
      throw new NotFoundException("The host named " + host + " was not found");
    }
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#registerHost(String, String, String, long, int, float)
   */
  @Override
  public void registerHost(String host, String address, String nodeName, long memory, int cores, float maxLoad)
          throws ServiceRegistryException {
    HostRegistrationInMemory hrim = new HostRegistrationInMemory(address, address, nodeName, maxLoad, cores, memory);
    hosts.put(host, hrim);
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#unregisterHost(java.lang.String)
   */
  @Override
  public void unregisterHost(String host) throws ServiceRegistryException {
    hosts.remove(host);
    services.remove(host);
  }

  /**
   * Method to register locally running services.
   *
   * @param localService
   *          the service instance
   * @return the service registration
   * @throws ServiceRegistryException
   */
  public ServiceRegistration registerService(JobProducer localService) throws ServiceRegistryException {
    return registerService(localService, Runtime.getRuntime().availableProcessors());
  }

  /**
   * Method to register locally running services.
   *
   * @param localService
   *          the service instance
   * @param maxLoad
   *          the maximum load the host can support
   * @return the service registration
   * @throws ServiceRegistryException
   */
  public ServiceRegistration registerService(JobProducer localService, float maxLoad) throws ServiceRegistryException {
    HostRegistrationInMemory hrim = hosts.get(LOCALHOST);

    List servicesOnHost = services.get(LOCALHOST);
    if (servicesOnHost == null) {
      servicesOnHost = new ArrayList();
      services.put(LOCALHOST, servicesOnHost);
    }

    ServiceRegistrationInMemoryImpl registration = new ServiceRegistrationInMemoryImpl(localService, hrim.getBaseUrl());
    registration.setMaintenance(false);
    servicesOnHost.add(registration);
    return registration;
  }

  /**
   * Removes the job producer from the service registry.
   *
   * @param localService
   *          the service
   * @throws ServiceRegistryException
   *           if removing the service fails
   */
  public void unregisterService(JobProducer localService) throws ServiceRegistryException {
    List servicesOnHost = services.get(LOCALHOST);
    if (servicesOnHost != null) {
      ServiceRegistrationInMemoryImpl s = (ServiceRegistrationInMemoryImpl) localService;
      servicesOnHost.remove(s);
    }
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#registerService(java.lang.String, java.lang.String,
   *      java.lang.String)
   */
  @Override
  public ServiceRegistration registerService(String serviceType, String host, String path)
          throws ServiceRegistryException {
    return registerService(serviceType, host, path, false);
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#registerService(java.lang.String, java.lang.String,
   *      java.lang.String, boolean)
   */
  @Override
  public ServiceRegistration registerService(String serviceType, String host, String path, boolean jobProducer)
          throws ServiceRegistryException {

    HostRegistrationInMemory hostRegistration = hosts.get(host);
    if (hostRegistration == null) {
      throw new ServiceRegistryException(new NotFoundException("Host " + host + " was not found"));
    }

    List servicesOnHost = services.get(host);
    if (servicesOnHost == null) {
      servicesOnHost = new ArrayList();
      services.put(host, servicesOnHost);
    }

    ServiceRegistrationInMemoryImpl registration = new ServiceRegistrationInMemoryImpl(serviceType, host, path,
            jobProducer);
    servicesOnHost.add(registration);
    return registration;
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#unRegisterService(java.lang.String, java.lang.String)
   */
  @Override
  public void unRegisterService(String serviceType, String host) throws ServiceRegistryException {
    List servicesOnHost = services.get(host);
    if (servicesOnHost != null) {
      Iterator ri = servicesOnHost.iterator();
      while (ri.hasNext()) {
        ServiceRegistration registration = ri.next();
        if (serviceType.equals(registration.getServiceType()))
          ri.remove();
      }
    }
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#setMaintenanceStatus(java.lang.String, boolean)
   */
  @Override
  public void setMaintenanceStatus(String host, boolean maintenance) throws NotFoundException {
    List servicesOnHost = services.get(host);
    if (!hosts.containsKey(host)) {
      throw new NotFoundException("Host " + host + " was not found");
    }
    hosts.get(host).setMaintenanceMode(maintenance);
    if (servicesOnHost != null) {
      for (ServiceRegistrationInMemoryImpl r : servicesOnHost) {
        r.setMaintenance(maintenance);
      }
    }
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#createJob(java.lang.String, java.lang.String)
   */
  @Override
  public Job createJob(String type, String operation) throws ServiceRegistryException {
    return createJob(type, operation, null, null, true);
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#createJob(java.lang.String, java.lang.String,
   *      java.util.List)
   */
  @Override
  public Job createJob(String type, String operation, List arguments) throws ServiceRegistryException {
    return createJob(type, operation, arguments, null, true);
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#createJob(java.lang.String, java.lang.String,
          java.util.List, Float)
   */
  @Override
  public Job createJob(String type, String operation, List arguments, Float jobLoad)
          throws ServiceRegistryException {
    return createJob(type, operation, arguments, null, true, jobLoad);
  }

  public Job createJob(String type, String operation, List arguments, String payload)
          throws ServiceRegistryException {
    return createJob(type, operation, arguments, payload, true);
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#createJob(java.lang.String, java.lang.String,
          java.util.List, java.lang.String, boolean)
   */
  @Override
  public Job createJob(String type, String operation, List arguments, String payload, boolean queueable)
          throws ServiceRegistryException {
    return createJob(type, operation, arguments, payload, queueable, null, 1.0f);
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#createJob(java.lang.String, java.lang.String,
          java.util.List, java.lang.String, boolean, Float)
   */
  @Override
  public Job createJob(String type, String operation, List arguments, String payload, boolean queueable,
          Float jobLoad) throws ServiceRegistryException {
    return createJob(type, operation, arguments, payload, queueable, null, jobLoad);
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#createJob(java.lang.String, java.lang.String,
          java.util.List, java.lang.String, boolean, org.opencastproject.job.api.Job, Float)
   */
  @Override
  public Job createJob(String type, String operation, List arguments, String payload, boolean queueable,
          Job parentJob, Float jobLoad) throws ServiceRegistryException {
    if (getServiceRegistrationsByType(type).size() == 0)
      logger.warn("Service " + type + " not available");

    Job job = null;
    synchronized (this) {
      job = new JobImpl(idCounter.addAndGet(1));
      if (securityService != null) {
        job.setCreator(securityService.getUser().getUsername());
        job.setOrganization(securityService.getOrganization().getId());
      }
      job.setDateCreated(new Date());
      job.setJobType(type);
      job.setOperation(operation);
      job.setArguments(arguments);
      job.setPayload(payload);
      if (queueable)
        job.setStatus(Status.QUEUED);
      else
        job.setStatus(Status.INSTANTIATED);
      if (parentJob != null)
        job.setParentJobId(parentJob.getId());
      job.setJobLoad(jobLoad);
    }

    synchronized (jobs) {
      try {
        jobs.put(job.getId(), JobParser.toXml(new JaxbJob(job)));
      } catch (IOException e) {
        throw new IllegalStateException("Error serializing job " + job, e);
      }
    }
    return job;
  }

  private void removeJob(long id) throws NotFoundException, ServiceRegistryException {
    synchronized (jobs) {
      if (!jobs.containsKey(id))
        throw new NotFoundException("No job with ID '" + id + "' found");

      jobs.remove(id);
    }
  }

  @Override
  public void removeJobs(List ids) throws NotFoundException, ServiceRegistryException {
    synchronized (jobs) {
      for (long id : ids) {
        removeJob(id);
      }
    }
  }

  /**
   * Dispatches the job to the least loaded service or throws a ServiceUnavailableException if there is no
   * such service.
   *
   * @param job
   *          the job to dispatch
   * @return whether the job was dispatched
   * @throws ServiceUnavailableException
   *           if no service is available to dispatch the job
   * @throws ServiceRegistryException
   *           if the service registrations are unavailable or dispatching of the job fails
   */
  protected boolean dispatchJob(Job job) throws ServiceUnavailableException, ServiceRegistryException,
          UndispatchableJobException {
    List registrations = getServiceRegistrationsByLoad(job.getJobType());
    if (registrations.size() == 0)
      throw new ServiceUnavailableException("No service is available to handle jobs of type '" + job.getJobType() + "'");
    job.setStatus(Status.DISPATCHING);
    try {
      job = updateJob(job);
    } catch (NotFoundException e) {
      throw new ServiceRegistryException("Job not found!", e);
    }
    for (ServiceRegistration registration : registrations) {
      if (registration.isJobProducer() && !registration.isInMaintenanceMode()) {
        ServiceRegistrationInMemoryImpl inMemoryRegistration = (ServiceRegistrationInMemoryImpl) registration;
        JobProducer service = inMemoryRegistration.getService();

        // Add the job to the list of jobs so that it gets counted in the load.
        // This is the same way that the JPA impl does it
        Set jobs = jobHosts.get(inMemoryRegistration);
        if (jobs == null) {
          jobs = new LinkedHashSet();
        }
        jobs.add(job);
        jobHosts.put(inMemoryRegistration, jobs);

        if (!service.isReadyToAcceptJobs(job.getOperation())) {
          jobs.remove(job);
          jobHosts.put(inMemoryRegistration, jobs);
          continue;
        }
        if (!service.isReadyToAccept(job)) {
          jobs.remove(job);
          jobHosts.put(inMemoryRegistration, jobs);
          continue;
        }
        try {
          job = updateJob(job);
        } catch (NotFoundException e) {
          jobs.remove(job);
          jobHosts.put(inMemoryRegistration, jobs);
          throw new ServiceRegistryException("Job not found!", e);
        }
        service.acceptJob(job);
        return true;
      } else if (!registration.isJobProducer()) {
        logger.warn("This implementation of the service registry doesn't support dispatching to remote services");
        // TODO: Add remote dispatching
      } else {
        logger.warn("Service " + registration + " is in maintenance mode");
      }
    }
    return false;
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#updateJob(org.opencastproject.job.api.Job)
   */
  @Override
  public Job updateJob(Job job) throws NotFoundException, ServiceRegistryException {
    if (job == null)
      throw new IllegalArgumentException("Job cannot be null");
    Job updatedJob = null;
    synchronized (jobs) {
      try {
        updatedJob = updateInternal(job);
        jobs.put(updatedJob.getId(), JobParser.toXml(new JaxbJob(updatedJob)));
      } catch (IOException e) {
        throw new IllegalStateException("Error serializing job", e);
      }
    }
    return updatedJob;
  }

  private Job updateInternal(Job job) {
    Date now = new Date();
    Status status = job.getStatus();
    if (job.getDateCreated() == null) {
      job.setDateCreated(now);
    }
    if (Status.RUNNING.equals(status)) {
      if (job.getDateStarted() == null) {
        job.setDateStarted(now);
        job.setQueueTime(now.getTime() - job.getDateCreated().getTime());
      }
    } else if (Status.FAILED.equals(status)) {
      // failed jobs may not have even started properly
      job.setDateCompleted(now);
      if (job.getDateStarted() != null) {
        job.setRunTime(now.getTime() - job.getDateStarted().getTime());
      }
    } else if (Status.FINISHED.equals(status)) {
      if (job.getDateStarted() == null) {
        // Some services (e.g. ingest) don't use job dispatching, since they start immediately and handle their own
        // lifecycle. In these cases, if the start date isn't set, use the date created as the start date
        job.setDateStarted(job.getDateCreated());
      }
      job.setDateCompleted(now);
      job.setRunTime(now.getTime() - job.getDateStarted().getTime());

      // Cleanup local list of jobs assigned to a specific service
      for (Entry> service : services.entrySet()) {
        for (ServiceRegistrationInMemoryImpl srv : service.getValue()) {
          Set jobs = jobHosts.get(srv);
          if (jobs != null) {
            Set updatedJobs = new HashSet<>();
            for (Job savedJob : jobs) {
              if (savedJob.getId() != job.getId())
                updatedJobs.add(savedJob);
            }
            jobHosts.put(srv, updatedJobs);
          }
        }
      }
    }
    return job;
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#getJob(long)
   */
  @Override
  public Job getJob(long id) throws NotFoundException, ServiceRegistryException {
    synchronized (jobs) {
      String serializedJob = jobs.get(id);
      if (serializedJob == null)
        throw new NotFoundException(Long.toString(id));
      try {
        return JobParser.parseJob(serializedJob);
      } catch (IOException e) {
        throw new IllegalStateException("Error unmarshaling job", e);
      }
    }
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#getChildJobs(long)
   */
  @Override
  public List getChildJobs(long id) throws ServiceRegistryException {
    List result = new ArrayList();
    synchronized (jobs) {
      for (String serializedJob : jobs.values()) {
        Job job = null;
        try {
          job = JobParser.parseJob(serializedJob);
        } catch (IOException e) {
          throw new IllegalStateException("Error unmarshaling job", e);
        }
        if (job.getParentJobId() == null)
          continue;
        if (job.getParentJobId().equals(id) || job.getRootJobId().equals(id))
          result.add(job);

        Long parentJobId = job.getParentJobId();
        while (parentJobId != null && parentJobId > 0) {
          try {
            Job parentJob = getJob(job.getParentJobId());
            if (parentJob.getParentJobId().equals(id)) {
              result.add(job);
              break;
            }
            parentJobId = parentJob.getParentJobId();
          } catch (NotFoundException e) {
            throw new ServiceRegistryException("Job from parent job id was not found!", e);
          }
        }
      }
    }
    Collections.sort(result, new Comparator() {
      @Override
      public int compare(Job job1, Job job2) {
        return job1.getDateCreated().compareTo(job1.getDateCreated());
      }
    });
    return result;
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#getJobs(java.lang.String,
   *      org.opencastproject.job.api.Job.Status)
   */
  @Override
  public List getJobs(String serviceType, Status status) throws ServiceRegistryException {
    List result = new ArrayList();
    synchronized (jobs) {
      for (String serializedJob : jobs.values()) {
        Job job = null;
        try {
          job = JobParser.parseJob(serializedJob);
        } catch (IOException e) {
          throw new IllegalStateException("Error unmarshaling job", e);
        }
        if (serviceType.equals(job.getJobType()) && status.equals(job.getStatus()))
          result.add(job);
      }
    }
    return result;
  }

  @Override
  public List getJobPayloads(String operation) throws ServiceRegistryException {
    List result = new ArrayList<>();
    for (String serializedJob : jobs.values()) {
      try {
        Job job = JobParser.parseJob(serializedJob);
        if (operation.equals(job.getOperation())) {
          result.add(job.getPayload());
        }
      } catch (IOException e) {
        throw new IllegalStateException("Error unmarshaling job", e);
      }
    }
    return result;
  }

  @Override
  public List getJobPayloads(String operation, int limit, int offset) throws ServiceRegistryException {
    return null;
  }

  @Override
  public int getJobCount(String operation) throws ServiceRegistryException {
    return 0;
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#getActiveJobs()
   */
  @Override
  public List getActiveJobs() throws ServiceRegistryException {
    List result = new ArrayList();
    synchronized (jobs) {
      for (String serializedJob : jobs.values()) {
        Job job = null;
        try {
          job = JobParser.parseJob(serializedJob);
        } catch (IOException e) {
          throw new IllegalStateException("Error unmarshaling job", e);
        }
        if (job.getStatus().isActive())
          result.add(job);
      }
    }
    return result;
  }

  @Override
  public Incidents incident() {
    return incidents;
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#getServiceRegistrationsByLoad(java.lang.String)
   */
  @Override
  public List getServiceRegistrationsByLoad(String serviceType) throws ServiceRegistryException {
    return getServiceRegistrationsByType(serviceType);
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#getServiceRegistrationsByType(java.lang.String)
   */
  @Override
  public List getServiceRegistrationsByType(String serviceType) throws ServiceRegistryException {
    List result = new ArrayList();
    for (List servicesPerHost : services.values()) {
      for (ServiceRegistrationInMemoryImpl r : servicesPerHost) {
        if (serviceType.equals(r.getServiceType()))
          result.add(r);
      }
    }
    return result;
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#getServiceRegistrationsByHost(java.lang.String)
   */
  @Override
  public List getServiceRegistrationsByHost(String host) throws ServiceRegistryException {
    List result = new ArrayList();
    List servicesPerHost = services.get(host);
    if (servicesPerHost != null) {
      result.addAll(servicesPerHost);
    }
    return result;
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#getServiceRegistration(java.lang.String,
   *      java.lang.String)
   */
  @Override
  public ServiceRegistration getServiceRegistration(String serviceType, String host) throws ServiceRegistryException {
    List servicesPerHost = services.get(host);
    if (servicesPerHost != null) {
      for (ServiceRegistrationInMemoryImpl r : servicesPerHost) {
        if (serviceType.equals(r.getServiceType()))
          return r;
      }
    }
    return null;
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#getServiceRegistrations()
   */
  @Override
  public List getServiceRegistrations() throws ServiceRegistryException {
    List result = new ArrayList();
    for (List servicesPerHost : services.values()) {
      result.addAll(servicesPerHost);
    }
    return result;
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#getServiceStatistics()
   */
  @Override
  public List getServiceStatistics() throws ServiceRegistryException {
    throw new UnsupportedOperationException("Operation not yet implemented");
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#count(java.lang.String,
   *      org.opencastproject.job.api.Job.Status)
   */
  @Override
  public long count(String serviceType, Status status) throws ServiceRegistryException {
    return count(serviceType, null, null, status);
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#countByOperation(java.lang.String, java.lang.String,
   *      org.opencastproject.job.api.Job.Status)
   */
  @Override
  public long countByOperation(String serviceType, String operation, Status status) throws ServiceRegistryException {
    return count(serviceType, null, operation, status);
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#countByHost(java.lang.String, java.lang.String,
   *      org.opencastproject.job.api.Job.Status)
   */
  @Override
  public long countByHost(String serviceType, String host, Status status) throws ServiceRegistryException {
    return count(serviceType, host, null, status);
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#count(java.lang.String, java.lang.String,
   *      java.lang.String, org.opencastproject.job.api.Job.Status)
   */
  @Override
  public long count(String serviceType, String host, String operation, Status status) throws ServiceRegistryException {
    int count = 0;
    synchronized (jobs) {
      for (String serializedJob : jobs.values()) {
        Job job = null;
        try {
          job = JobParser.parseJob(serializedJob);
        } catch (IOException e) {
          throw new IllegalStateException("Error unmarshaling job", e);
        }
        if (serviceType != null && !serviceType.equals(job.getJobType()))
          continue;
        if (host != null && !host.equals(job.getProcessingHost()))
          continue;
        if (operation != null && !operation.equals(job.getOperation()))
          continue;
        if (status != null && !status.equals(job.getStatus()))
          continue;
        count++;
      }
    }
    return count;
  }

  /**
   * This dispatcher implementation will wake from time to time and check for new jobs. If new jobs are found, it will
   * dispatch them to the services as appropriate.
   */
  class JobDispatcher implements Runnable {

    /**
     * {@inheritDoc}
     *
     * @see java.lang.Thread#run()
     */
    @Override
    public void run() {

      // Go through the jobs and find those that have not yet been dispatched
      synchronized (jobs) {
        for (String serializedJob : jobs.values()) {
          Job job = null;
          try {
            job = JobParser.parseJob(serializedJob);
            User creator = userDirectoryService.loadUser(job.getCreator());
            Organization organization = organizationDirectoryService.getOrganization(job.getOrganization());
            securityService.setUser(creator);
            securityService.setOrganization(organization);
            if (Status.QUEUED.equals(job.getStatus())) {
              job.setStatus(Status.DISPATCHING);
              if (!dispatchJob(job)) {
                job.setStatus(Status.QUEUED);
              }
            }
          } catch (ServiceUnavailableException e) {
            job.setStatus(Status.FAILED);
            Throwable cause = (e.getCause() != null) ? e.getCause() : e;
            logger.error("Unable to find a service for job " + job, cause);
          } catch (ServiceRegistryException e) {
            job.setStatus(Status.FAILED);
            Throwable cause = (e.getCause() != null) ? e.getCause() : e;
            logger.error("Error dispatching job " + job, cause);
          } catch (IOException e) {
            throw new IllegalStateException("Error unmarshaling job", e);
          } catch (NotFoundException e) {
            throw new IllegalStateException("Creator organization not found", e);
          } catch (Throwable e) {
            logger.error("Error dispatching job " + job, e);
          } finally {
            try {
              jobs.put(job.getId(), JobParser.toXml(new JaxbJob(job)));
            } catch (IOException e) {
              throw new IllegalStateException("Error unmarshaling job", e);
            }
            securityService.setUser(null);
            securityService.setOrganization(null);
          }
        }
      }
    }
  }

  /** Shuts down this service registry, logging all jobs and their statuses. */
  public void deactivate() {
    dispatcher.shutdownNow();
    Map counts = new HashMap();
    synchronized (jobs) {
      for (String serializedJob : jobs.values()) {
        Job job = null;
        try {
          job = JobParser.parseJob(serializedJob);
        } catch (IOException e) {
          throw new IllegalStateException("Error unmarshaling job", e);
        }
        if (counts.containsKey(job.getStatus())) {
          counts.get(job.getStatus()).incrementAndGet();
        } else {
          counts.put(job.getStatus(), new AtomicInteger(1));
        }
      }
    }
    StringBuilder sb = new StringBuilder("Abandoned:");
    for (Entry entry : counts.entrySet()) {
      sb.append(" " + entry.getValue() + " " + entry.getKey() + " jobs");
    }
    logger.info(sb.toString());
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#getMaxLoads()
   */
  @Override
  public SystemLoad getMaxLoads() throws ServiceRegistryException {
    SystemLoad systemLoad = new SystemLoad();
    systemLoad.addNodeLoad(new NodeLoad(LOCALHOST, 0.0f, Runtime.getRuntime().availableProcessors()));
    return systemLoad;
  }

  /**
   * {@inheritDoc}
   *
   * @see org.opencastproject.serviceregistry.api.ServiceRegistry#getMaxLoadOnNode(java.lang.String)
   */
  @Override
  public NodeLoad getMaxLoadOnNode(String host) throws ServiceRegistryException {
    if (hosts.containsKey(host)) {
      return new NodeLoad(host, 0.0f, hosts.get(host).getMaxLoad());
    }
    throw new ServiceRegistryException("Unable to find host " + host + " in service registry");
  }

  /**
   * Sets the security service.
   *
   * @param securityService
   *          the securityService to set
   */
  public void setSecurityService(SecurityService securityService) {
    this.securityService = securityService;
  }

  @Override
  public void sanitize(String serviceType, String host) {
    // TODO Auto-generated method stub
  }

  @Override
  public Job getCurrentJob() {
    return this.currentJob;
  }

  @Override
  public void setCurrentJob(Job job) {
    this.currentJob = job;
  }

  @Override
  public List getHostRegistrations() throws ServiceRegistryException {
    List hostList = new LinkedList();
    hostList.addAll(hosts.values());
    return hostList;
  }

  @Override
  public HostStatistics getHostStatistics() {
    HostStatistics statistics = new HostStatistics();
    for (Map.Entry> entry: jobHosts.entrySet()) {
      final ServiceRegistrationInMemoryImpl service = entry.getKey();
      final long queued = entry.getValue().stream().filter(job -> job.getStatus() == Status.QUEUED).count();
      final long running = entry.getValue().stream().filter(job -> job.getStatus() == Status.RUNNING).count();
      final long host = service.host.hashCode();
      statistics.addQueued(host, statistics.queuedJobs(host) + queued);
      statistics.addRunning(host, statistics.runningJobs(host) + running);
    }
    return statistics;
  }

  @Override
  public HostRegistration getHostRegistration(String hostname) throws ServiceRegistryException {
    for (HostRegistration host:  this.getHostRegistrations()) {
      if (host.getBaseUrl().equalsIgnoreCase(hostname)) {
        return host;
      }
    }
    throw new ServiceRegistryException(String.format("Host registration for %s not found", hostname));
  }

  @Override
  public SystemLoad getCurrentHostLoads() {
    SystemLoad systemLoad = new SystemLoad();

    for (String host : hosts.keySet()) {
      NodeLoad node = new NodeLoad();
      node.setHost(host);
      for (ServiceRegistration service : services.get(host)) {
        if (service.isInMaintenanceMode() || !service.isOnline()) {
          continue;
        }
        Set hostJobs = jobHosts.get(service);
        float loadSum = 0.0f;
        if (hostJobs != null) {
          for (Job job : hostJobs) {
            if (job.getStatus() != null && JOB_STATUSES_INFLUENCING_LOAD_BALANCING.contains(job.getStatus())) {
              loadSum += job.getJobLoad();
            }
          }
        }
        node.setCurrentLoad(loadSum);
      }
      systemLoad.addNodeLoad(node);
    }
    return systemLoad;
  }

  @Override
  public void removeParentlessJobs(int lifetime) throws ServiceRegistryException {
    synchronized (jobs) {
      for (String serializedJob : jobs.values()) {
        Job job = null;
        try {
          job = JobParser.parseJob(serializedJob);
        } catch (IOException e) {
          throw new IllegalStateException("Error unmarshaling job", e);
        }

        Long parentJobId = job.getParentJobId();
        if (parentJobId == null | parentJobId < 1)
          jobs.remove(job.getId());
      }
    }
  }

  @Override
  public float getOwnLoad() {
    return getCurrentHostLoads().get(getRegistryHostname()).getCurrentLoad();
  }

  @Override
  public String getRegistryHostname() {
    return LOCALHOST;
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy