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

org.opencastproject.scheduler.impl.SchedulerUtil 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.scheduler.impl;

import static com.entwinemedia.fn.Prelude.chuck;
import static com.entwinemedia.fn.Stream.$;

import org.opencastproject.mediapackage.Catalog;
import org.opencastproject.mediapackage.EName;
import org.opencastproject.mediapackage.MediaPackage;
import org.opencastproject.mediapackage.MediaPackageElementFlavor;
import org.opencastproject.mediapackage.MediaPackageElements;
import org.opencastproject.mediapackage.MediaPackageSupport;
import org.opencastproject.metadata.dublincore.DublinCore;
import org.opencastproject.metadata.dublincore.DublinCoreCatalog;
import org.opencastproject.metadata.dublincore.DublinCoreUtil;
import org.opencastproject.metadata.dublincore.DublinCoreValue;
import org.opencastproject.metadata.dublincore.EventCatalogUIAdapter;
import org.opencastproject.scheduler.api.SchedulerEvent;
import org.opencastproject.scheduler.api.TechnicalMetadata;
import org.opencastproject.security.api.AccessControlList;
import org.opencastproject.security.api.AccessControlUtil;
import org.opencastproject.util.Checksum;
import org.opencastproject.util.DateTimeSupport;
import org.opencastproject.workspace.api.Workspace;

import com.entwinemedia.fn.Fn;
import com.entwinemedia.fn.Fn2;
import com.entwinemedia.fn.data.Opt;

import org.apache.commons.lang3.CharUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.nio.charset.StandardCharsets;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.Date;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
import java.util.TreeMap;

/**
 * Functions to support scheduler service operations.
 */
public final class SchedulerUtil {

  /** The logger */
  private static final Logger logger = LoggerFactory.getLogger(SchedulerUtil.class);

  private SchedulerUtil() {
  }

  public static final Comparator sortCatalogById = new Comparator() {
    @Override
    public int compare(Catalog c1, Catalog c2) {
      return c1.getIdentifier().compareTo(c2.getIdentifier());
    }
  };

  public static String calculateChecksum(
      Workspace workspace,
      List eventCatalogUIAdapterFlavors,
      Date startDateTime,
      Date endDateTime,
      String captureAgentId,
      Set userIds,
      MediaPackage mediaPackage,
      Opt episodeDublincore,
      Map wfProperties,
      Map finalCaProperties,
      AccessControlList acl) {
    List userIdsList = new ArrayList<>(userIds);
    Collections.sort(userIdsList);
    final MessageDigest messageDigest = mkMd5MessageDigest();
    messageDigest.update(mkChecksumInput(startDateTime));
    messageDigest.update(mkChecksumInput(endDateTime));
    messageDigest.update(mkChecksumInput(captureAgentId));
    for (String user : userIdsList) {
      messageDigest.update(mkChecksumInput(user));
    }
    if (episodeDublincore.isSome()) {
      Catalog episodeCatalog = $(mediaPackage.getCatalogs())
          .filter(MediaPackageSupport.Filters.isEpisodeDublinCore.toFn()).head2();
      Checksum checksum = episodeCatalog.getChecksum();
      if (checksum == null) {
        checksum = DublinCoreUtil.calculateChecksum(episodeDublincore.get());
        episodeCatalog.setChecksum(checksum);
      }
      messageDigest.update(mkChecksumInput(checksum.toString()));
    }
    // Add extended metadata to calculation
    for (Catalog c : $(mediaPackage.getCatalogs()).sort(sortCatalogById)) {
      if (eventCatalogUIAdapterFlavors.contains(c.getFlavor())) {
        Checksum checksum = c.getChecksum();
        if (checksum == null) {
          DublinCoreCatalog dublinCore = DublinCoreUtil.loadDublinCore(workspace, c);
          checksum = DublinCoreUtil.calculateChecksum(dublinCore);
          c.setChecksum(checksum);
        }
        messageDigest.update(mkChecksumInput(checksum.toString()));
      }
    }
    messageDigest.update(mkChecksumInput(AccessControlUtil.calculateChecksum(acl).toString()));
    for (Entry entry : new TreeMap<>(wfProperties).entrySet()) {
      messageDigest.update(mkChecksumInput(entry.getKey()));
      messageDigest.update(mkChecksumInput(entry.getValue()));
    }
    for (Entry entry : new TreeMap<>(finalCaProperties).entrySet()) {
      messageDigest.update(mkChecksumInput(entry.getKey()));
      messageDigest.update(mkChecksumInput(entry.getValue()));
    }
    return Checksum.convertToHex(messageDigest.digest());
  }

  private static MessageDigest mkMd5MessageDigest() {
    try {
      return MessageDigest.getInstance("MD5");
    } catch (NoSuchAlgorithmException e) {
      logger.error("Unable to create md5 message digest");
      return chuck(e);
    }
  }

  private static byte[] mkChecksumInput(String input) {
    return input.getBytes(StandardCharsets.UTF_8);
  }

  private static byte[] mkChecksumInput(Date input) {
    return mkChecksumInput(Long.toString(input.getTime()));
  }

  /**
   * Converts a scheduler event to a human readable string
   *
   * @param workspace
   *          the workspace
   * @param catalogFlavors
   *          the event catalog flavors
   * @param event
   *          the scheduler event
   * @return a human readable string
   */
  public static String toHumanReadableString(Workspace workspace, List catalogFlavors,
          SchedulerEvent event) {
    TechnicalMetadata technicalMetadata = event.getTechnicalMetadata();
    StringBuilder sb = new StringBuilder("Event: ").append(CharUtils.LF);
    sb.append("- ").append(event.getEventId()).append(CharUtils.LF);
    sb.append(CharUtils.LF);

    sb.append("Version").append(CharUtils.LF);
    sb.append("- ").append(event.getVersion()).append(CharUtils.LF);
    sb.append(CharUtils.LF);

    sb.append("Start").append(CharUtils.LF);
    sb.append("- ").append(DateTimeSupport.toUTC(technicalMetadata.getStartDate().getTime())).append(CharUtils.LF);
    sb.append(CharUtils.LF);

    sb.append("End").append(CharUtils.LF);
    sb.append("- ").append(DateTimeSupport.toUTC(technicalMetadata.getEndDate().getTime())).append(CharUtils.LF);
    sb.append(CharUtils.LF);

    sb.append("Room").append(CharUtils.LF);
    sb.append("- ").append(technicalMetadata.getAgentId()).append(CharUtils.LF);
    sb.append(CharUtils.LF);

    sb.append("Scheduling configuration").append(CharUtils.LF);
    for (Entry entry : technicalMetadata.getCaptureAgentConfiguration().entrySet()) {
      sb.append("- ").append(entry.getKey()).append(": ").append(entry.getValue()).append(CharUtils.LF);
    }
    sb.append(CharUtils.LF);

    sb.append("Presenters").append(CharUtils.LF);
    for (String presenter : technicalMetadata.getPresenters()) {
      sb.append("- ").append(presenter).append(CharUtils.LF);
    }
    sb.append(CharUtils.LF);

    sb.append("Workflow configuration").append(CharUtils.LF);
    for (Entry entry : technicalMetadata.getWorkflowProperties().entrySet()) {
      sb.append("- ").append(entry.getKey()).append(": ").append(entry.getValue()).append(CharUtils.LF);
    }
    sb.append(CharUtils.LF);

    for (Catalog c : $(event.getMediaPackage().getCatalogs())) {
      if (!catalogFlavors.contains(c.getFlavor()))
        continue;

      DublinCoreCatalog dublinCore;
      try {
        dublinCore = DublinCoreUtil.loadDublinCore(workspace, c);
      } catch (Exception e) {
        logger.error("Unable to read event dublincore", e);
        continue;
      }

      sb.append("Event metadata ").append("(").append(c.getFlavor().toString()).append(")").append(CharUtils.LF);
      for (Entry> entry : dublinCore.getValues().entrySet()) {
        EName eName = entry.getKey();
        for (DublinCoreValue value : entry.getValue()) {
          sb.append("- ").append(eName.getNamespaceURI()).append(":").append(eName.getLocalName()).append(": ")
                  .append(value.getValue());

          boolean hasLanguageDefined = !DublinCore.LANGUAGE_UNDEFINED.equals(value.getLanguage());

          if (hasLanguageDefined || value.getEncodingScheme().isSome()) {
            sb.append(" (");
            if (hasLanguageDefined) {
              sb.append("lang:").append(value.getLanguage());
              if (value.getEncodingScheme().isSome())
                sb.append("/");
            }

            for (EName schema : value.getEncodingScheme()) {
              sb.append(schema.getLocalName());
            }
            sb.append(")");
          }
          sb.append(CharUtils.LF);
        }
      }
      sb.append(CharUtils.LF);
    }
    return sb.toString();
  }

  public static final Fn isNotEpisodeDublinCore = new Fn() {
    @Override
    public Boolean apply(MediaPackageElementFlavor mpe) {
      // match is commutative
      return !MediaPackageElements.EPISODE.matches(mpe);
    }
  };

  public static final Fn uiAdapterToFlavor = new Fn() {
    @Override
    public MediaPackageElementFlavor apply(EventCatalogUIAdapter adapter) {
      return adapter.getFlavor();
    }
  };

  public static final Fn2 eventOrganizationFilter = new Fn2() {
    @Override
    public Boolean apply(EventCatalogUIAdapter catalogUIAdapter, String organization) {
      return catalogUIAdapter.getOrganization().equals(organization);
    }
  };
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy