OphaalConfigActionBean.java

/*
 * Copyright (C) 2015 B3Partners B.V.
 */
package nl.b3p.brmo.service.stripes;

import static nl.b3p.brmo.service.jobs.GeplandeTakenInit.QUARTZ_FACTORY_KEY;
import static nl.b3p.brmo.service.jobs.GeplandeTakenInit.SCHEDULER_NAME;

import java.text.ParseException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import net.sourceforge.stripes.action.ActionBean;
import net.sourceforge.stripes.action.ActionBeanContext;
import net.sourceforge.stripes.action.Before;
import net.sourceforge.stripes.action.DefaultHandler;
import net.sourceforge.stripes.action.DontValidate;
import net.sourceforge.stripes.action.ForwardResolution;
import net.sourceforge.stripes.action.RedirectResolution;
import net.sourceforge.stripes.action.Resolution;
import net.sourceforge.stripes.action.SimpleMessage;
import net.sourceforge.stripes.action.StrictBinding;
import net.sourceforge.stripes.controller.LifecycleStage;
import net.sourceforge.stripes.validation.SimpleError;
import net.sourceforge.stripes.validation.Validate;
import net.sourceforge.stripes.validation.ValidateNestedProperties;
import net.sourceforge.stripes.validation.ValidationMethod;
import nl.b3p.brmo.persistence.staging.AfgifteNummerScannerProces;
import nl.b3p.brmo.persistence.staging.AutomatischProces;
import nl.b3p.brmo.persistence.staging.BAG2MutatieProces;
import nl.b3p.brmo.persistence.staging.BGTLoaderProces;
import nl.b3p.brmo.persistence.staging.BRK2ScannerProces;
import nl.b3p.brmo.persistence.staging.BerichtDoorstuurProces;
import nl.b3p.brmo.persistence.staging.BerichtTransformatieProces;
import nl.b3p.brmo.persistence.staging.BerichtstatusRapportProces;
import nl.b3p.brmo.persistence.staging.ClobElement;
import nl.b3p.brmo.persistence.staging.GDS2OphaalProces;
import nl.b3p.brmo.persistence.staging.LaadprocesStatusRapportProces;
import nl.b3p.brmo.persistence.staging.LaadprocesTransformatieProces;
import nl.b3p.brmo.persistence.staging.MailRapportageProces;
import nl.b3p.brmo.persistence.staging.MaterializedViewRefresh;
import nl.b3p.brmo.persistence.staging.TopNLScannerProces;
import nl.b3p.brmo.service.jobs.GeplandeTakenInit;
import nl.b3p.brmo.service.scanner.ProcesExecutable;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.quartz.CronExpression;
import org.quartz.JobKey;
import org.quartz.Scheduler;
import org.quartz.SchedulerException;
import org.quartz.impl.StdSchedulerFactory;
import org.stripesstuff.stripersist.EntityTypeConverter;
import org.stripesstuff.stripersist.Stripersist;

/**
 * @author mprins
 */
@StrictBinding
public class OphaalConfigActionBean implements ActionBean {

  private static final Log log = LogFactory.getLog(OphaalConfigActionBean.class);

  private static final String JSP = "/WEB-INF/jsp/beheer/processenophalen.jsp";

  private static final String DEFAULT_BAG2_MUTATIE_MIRROR =
      "https://bag.b3p.nl/dagmutaties/bestanden.json";

  private ActionBeanContext context;

  private List<AutomatischProces> processen = new ArrayList<>();

  @Validate(converter = EntityTypeConverter.class)
  @ValidateNestedProperties({@Validate(field = "cronExpressie", on = "save")})
  private AutomatischProces proces;

  @Validate private ProcesExecutable.ProcessingImple type;

  @Validate private Map<String, ClobElement> config = new HashMap<>();

  @Before(stages = LifecycleStage.BindingAndValidation)
  public void load() {
    processen =
        Stripersist.getEntityManager()
            .createQuery("from AutomatischProces p order by type(p), p.id", AutomatischProces.class)
            .getResultList();
  }

  @DefaultHandler
  public Resolution view() {
    if (proces != null) {
      config = proces.getConfig();
    }

    return new ForwardResolution(JSP);
  }

  @DontValidate
  public Resolution cancel() {
    return new RedirectResolution(OphaalConfigActionBean.class);
  }

  @DontValidate
  public Resolution delete() {
    if (proces != null) {

      Stripersist.getEntityManager()
          .createQuery(
              "update LaadProces set automatischProces = null where automatischProces = :this")
          .setParameter("this", proces)
          .executeUpdate();

      Stripersist.getEntityManager().remove(proces);
      load();
      Stripersist.getEntityManager().getTransaction().commit();
      deleteScheduledJob(proces);
      proces = null;
      getContext().getMessages().add(new SimpleMessage("Proces is verwijderd"));
    }
    return new ForwardResolution(JSP);
  }

  @DontValidate
  public Resolution add() {
    if (type == ProcesExecutable.ProcessingImple.BGTLoaderProces) {
      // default values voor BGT loader
      config.put("create-schema", new ClobElement("true"));
      config.put("feature-types", new ClobElement("all"));
    } else if (type == ProcesExecutable.ProcessingImple.BAG2MutatieProces) {
      config.put("url", new ClobElement(DEFAULT_BAG2_MUTATIE_MIRROR));
      proces = getProces(type);
      proces.setCronExpressie("0 30 2 * * ? *");
    }
    return new ForwardResolution(JSP).addParameter("type", type);
  }

  public Resolution save() {
    if (proces == null) {
      proces = getProces(type);
    }

    proces.getConfig().clear();
    proces.getConfig().putAll(config);

    if (proces instanceof BerichtDoorstuurProces) {
      String id = ClobElement.nullSafeGet(config.get("gds2_ophaalproces_id"));
      GDS2OphaalProces p =
          Stripersist.getEntityManager().find(GDS2OphaalProces.class, Long.parseLong(id));
      if (p != null) {
        String label = ClobElement.nullSafeGet(p.getConfig().get("label"));
        proces.getConfig().put("label", new ClobElement("Doorsturen " + label + " afgiftes"));
      }
    }

    Stripersist.getEntityManager().persist(proces);
    load();
    Stripersist.getEntityManager().getTransaction().commit();
    getContext().getMessages().add(new SimpleMessage("Proces is opgeslagen"));

    try {
      this.updateJobSchedule(proces);
    } catch (SchedulerException ex) {
      getContext()
          .getMessages()
          .add(
              new SimpleError(
                  "Er is een fout opgetreden tijdens inplannen van de taak. {2}", ex.getMessage()));
    }

    return new ForwardResolution(JSP).addParameter("proces", proces.getId());
  }

  /**
   * ProcesExecutable factory.
   *
   * @param type the type of {@code AutomatischProces} to create.
   * @return an instance of the specified type
   */
  private AutomatischProces getProces(ProcesExecutable.ProcessingImple type) {
    return switch (type) {
      case BAG2MutatieProces -> new BAG2MutatieProces();
      case BRK2ScannerProces -> new BRK2ScannerProces();
      case MailRapportageProces -> new MailRapportageProces();
      case GDS2OphaalProces -> new GDS2OphaalProces();
      case BerichtTransformatieProces -> new BerichtTransformatieProces();
      case BerichtDoorstuurProces -> new BerichtDoorstuurProces();
      case LaadprocesTransformatieProces -> new LaadprocesTransformatieProces();
      case MaterializedViewRefresh -> new MaterializedViewRefresh();
      case BerichtstatusRapportProces -> new BerichtstatusRapportProces();
      case LaadprocesStatusRapportProces -> new LaadprocesStatusRapportProces();
      case TopNLScannerProces -> new TopNLScannerProces();
      case AfgifteNummerScannerProces -> new AfgifteNummerScannerProces();
      case BGTLoaderProces -> new BGTLoaderProces();
      default ->
          throw new IllegalArgumentException(type.name() + " is geen ondersteund proces type...");
    };
  }

  private ProcesExecutable.ProcessingImple getType(AutomatischProces p) {
    return ProcesExecutable.ProcessingImple.valueOf(p.getClass().getSimpleName());
  }

  /**
   * Vervang de proces job door een aangepaste job of plaats een nieuwe job mocht deze nog niet
   * bestaan. Als de cron expressie {@code null} is wordt de job verwijderd uit de scheduler.
   *
   * @param p bij te werken proces
   * @throws SchedulerException bij een quartz fout
   */
  private void updateJobSchedule(AutomatischProces p) throws SchedulerException {
    log.debug("Update scheduled job:" + p.getId());

    StdSchedulerFactory factory =
        (StdSchedulerFactory) getContext().getServletContext().getAttribute(QUARTZ_FACTORY_KEY);
    Scheduler scheduler = factory.getScheduler(SCHEDULER_NAME);
    JobKey key = new JobKey(GeplandeTakenInit.JOBKEY_PREFIX + p.getId());
    log.debug("Jobkey voor id " + p.getId() + " gevonden? " + scheduler.checkExists(key));

    scheduler.deleteJob(key);
    if (p.getCronExpressie() != null) {
      GeplandeTakenInit.addJobDetails(scheduler, p);
    }
  }

  /**
   * verwijder een proces job uit de scheduler door update met een {@code null} cron schedule.
   *
   * @param p te verwijderen proces
   */
  private void deleteScheduledJob(AutomatischProces p) {
    try {
      p.setCronExpressie(null);
      updateJobSchedule(p);
    } catch (SchedulerException se) {
      log.warn("Ingeplande taak uit de planner halen is mislukt", se);
    }
  }

  /** validatie van de cron expressie van een proces voorafgaand aan save. */
  @ValidationMethod(on = "save")
  public void validateCronExpressie() {
    if (proces != null) {
      String expr = proces.getCronExpressie();
      try {
        if (expr != null) {
          CronExpression cron = new CronExpression(expr);
        }
      } catch (ParseException ex) {
        Stripersist.getEntityManager().refresh(proces);
        load();
        getContext()
            .getValidationErrors()
            .add(
                "cronExpressie",
                new SimpleError(
                    "{0} {2} is ongeldig, (melding: {4}, mogelijk nabij positie {3})",
                    expr, ex.getErrorOffset(), ex.getLocalizedMessage()));
      }
    }
  }

  // <editor-fold defaultstate="collapsed" desc="getters en setters">
  @Override
  public ActionBeanContext getContext() {
    return context;
  }

  @Override
  public void setContext(ActionBeanContext context) {
    this.context = context;
  }

  public List<AutomatischProces> getProcessen() {
    return processen;
  }

  public void setProcessen(List<AutomatischProces> processen) {
    this.processen = processen;
  }

  public AutomatischProces getProces() {
    return proces;
  }

  public void setProces(AutomatischProces proces) {
    this.proces = proces;
    this.type = getType(proces);
  }

  public Map<String, ClobElement> getConfig() {
    return config;
  }

  public void setConfig(Map<String, ClobElement> config) {
    this.config = config;
  }

  public ProcesExecutable.ProcessingImple getType() {
    return type;
  }

  public void setType(ProcesExecutable.ProcessingImple type) {
    this.type = type;
    this.proces = getProces(type);
  }
  // </editor-fold>
}