GDS2OphalenProces.java

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

import static nl.b3p.brmo.persistence.staging.AutomatischProces.LOG_NEWLINE;
import static nl.b3p.gds2.GDS2Util.*;

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileWriter;
import java.io.IOException;
import java.io.InputStream;
import java.io.PrintWriter;
import java.io.StringWriter;
import java.math.BigInteger;
import java.net.HttpURLConnection;
import java.net.URL;
import java.net.URLConnection;
import java.nio.charset.StandardCharsets;
import java.security.KeyManagementException;
import java.security.KeyStore;
import java.security.KeyStoreException;
import java.security.NoSuchAlgorithmException;
import java.security.NoSuchProviderException;
import java.security.PrivateKey;
import java.security.cert.Certificate;
import java.security.cert.CertificateException;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Calendar;
import java.util.Date;
import java.util.GregorianCalendar;
import java.util.List;
import java.util.Map;
import java.util.zip.ZipEntry;
import java.util.zip.ZipInputStream;
import javax.net.ssl.HttpsURLConnection;
import javax.net.ssl.KeyManagerFactory;
import javax.net.ssl.SSLContext;
import javax.net.ssl.TrustManager;
import javax.net.ssl.TrustManagerFactory;
import javax.net.ssl.X509TrustManager;
import javax.persistence.*;
import javax.xml.ws.BindingProvider;
import javax.xml.ws.handler.Handler;
import nl.b3p.brmo.loader.entity.Brk2Bericht;
import nl.b3p.brmo.loader.util.BrmoException;
import nl.b3p.brmo.loader.xml.Brk2SnapshotXMLReader;
import nl.b3p.brmo.persistence.staging.AutomatischProces;
import nl.b3p.brmo.persistence.staging.Bericht;
import nl.b3p.brmo.persistence.staging.ClobElement;
import nl.b3p.brmo.persistence.staging.GDS2OphaalProces;
import nl.b3p.brmo.persistence.staging.LaadProces;
import nl.b3p.brmo.service.util.TrustManagerDelegate;
import nl.b3p.brmo.soap.util.LogMessageHandler;
import nl.kadaster.schemas.gds2.afgifte_bestandenlijstopvragen.v20170401.BestandenlijstOpvragenType;
import nl.kadaster.schemas.gds2.afgifte_bestandenlijstresultaat.afgifte.v20170401.AfgifteType;
import nl.kadaster.schemas.gds2.afgifte_bestandenlijstselectie.v20170401.AfgifteSelectieCriteriaType;
import nl.kadaster.schemas.gds2.afgifte_bestandenlijstselectie.v20170401.BestandKenmerkenType;
import nl.kadaster.schemas.gds2.afgifte_proces.v20170401.FilterDatumTijdType;
import nl.kadaster.schemas.gds2.afgifte_proces.v20170401.KlantAfgiftenummerReeksType;
import nl.kadaster.schemas.gds2.imgds.baseurl.v20170401.BaseURLType;
import nl.kadaster.schemas.gds2.service.afgifte.v20170401.Gds2AfgifteServiceV20170401;
import nl.kadaster.schemas.gds2.service.afgifte.v20170401.Gds2AfgifteServiceV20170401Service;
import nl.kadaster.schemas.gds2.service.afgifte_bestandenlijstopvragen.v20170401.BestandenlijstOpvragenRequest;
import nl.kadaster.schemas.gds2.service.afgifte_bestandenlijstopvragen.v20170401.BestandenlijstOpvragenResponse;
import nl.logius.digikoppeling.gb._2010._10.DataReference;
import org.apache.commons.io.IOUtils;
import org.apache.commons.lang3.StringUtils;
import org.apache.commons.lang3.exception.ExceptionUtils;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.apache.cxf.configuration.jsse.TLSClientParameters;
import org.apache.cxf.endpoint.Client;
import org.apache.cxf.frontend.ClientProxy;
import org.apache.cxf.transport.http.HTTPConduit;
import org.stripesstuff.stripersist.Stripersist;

/**
 * @author Matthijs Laan
 */
public class GDS2OphalenProces extends AbstractExecutableProces {

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

  private final GDS2OphaalProces config;

  @Transient private ProgressUpdateListener l;

  @Transient private SSLContext context;

  private static final int BESTANDENLIJST_ATTEMPTS = 5;
  private static final int BESTANDENLIJST_RETRY_WAIT = 10000;

  private static final int AFGIFTE_DOWNLOAD_ATTEMPTS = 5;
  private static final int AFGIFTE_DOWNLOAD_RETRY_WAIT = 3000;

  public GDS2OphalenProces(GDS2OphaalProces config) {
    this.config = config;
  }

  public static Log getLog() {
    return log;
  }

  @Override
  public void execute() throws BrmoException {
    this.execute(
        new ProgressUpdateListener() {
          @Override
          public void total(long total) {}

          @Override
          public void progress(long progress) {}

          @Override
          public void exception(Throwable t) {
            log.error(t);
          }

          @Override
          public void updateStatus(String status) {}

          @Override
          public void addLog(String log) {
            GDS2OphalenProces.log.info(log);
          }
        });
  }

  @Override
  public void execute(ProgressUpdateListener listener) {
    this.l = listener;

    try {
      l.updateStatus("Initialiseren...");
      final Date startDate = new Date();
      l.addLog(
          String.format("Initialiseren GDS2 ophalen proces %d... %tc", config.getId(), startDate));
      this.config.updateSamenvattingEnLogfile(
          String.format("Het GDS2 ophalen proces is gestart op %tc.", startDate));
      this.config.setStatus(AutomatischProces.ProcessingStatus.PROCESSING);
      this.config.setLastrun(startDate);
      Stripersist.getEntityManager().merge(this.config);
      Stripersist.getEntityManager().flush();

      BestandenlijstOpvragenRequest request = new BestandenlijstOpvragenRequest();
      BestandenlijstOpvragenType verzoek = new BestandenlijstOpvragenType();
      AfgifteSelectieCriteriaType criteria = new AfgifteSelectieCriteriaType();
      request.setVerzoek(verzoek);
      verzoek.setAfgifteSelectieCriteria(criteria);
      criteria.setBestandKenmerken(new BestandKenmerkenType());

      String contractnummer =
          ClobElement.nullSafeGet(this.config.getConfig().get("gds2_contractnummer"));
      if (contractnummer != null) {
        log.debug("GDS2 contractnummer is: " + contractnummer);
        criteria.getBestandKenmerken().setContractnummer(contractnummer);
      }
      String artikelnummer =
          ClobElement.nullSafeGet(this.config.getConfig().get("gds2_artikelnummer"));
      if (artikelnummer != null) {
        log.debug("GDS2 artikelnummer is: " + artikelnummer);
        l.addLog("GDS2 artikelnummer is: " + artikelnummer);
        criteria.getBestandKenmerken().setArtikelnummer(artikelnummer);
      }

      String alGerapporteerd =
          ClobElement.nullSafeGet(this.config.getConfig().get("gds2_al_gerapporteerde_afgiftes"));
      if (!"true".equals(alGerapporteerd)) {
        criteria.setNogNietGerapporteerd(true);
      }

      // datum tijd parsen/instellen
      GregorianCalendar vanaf;
      GregorianCalendar tot =
          getDatumTijd(ClobElement.nullSafeGet(this.config.getConfig().get("totdatum")));
      String sVanaf = ClobElement.nullSafeGet(this.config.getConfig().get("vanafdatum"));
      if (tot != null && ("-1".equals(sVanaf) | "-2".equals(sVanaf) | "-3".equals(sVanaf))) {
        vanaf =
            getDatumTijd(
                ClobElement.nullSafeGet(this.config.getConfig().get("totdatum")),
                Integer.parseInt(sVanaf));
      } else {
        vanaf = getDatumTijd(sVanaf);
      }

      List<AfgifteType> afgiftes = new ArrayList<>();
      BestandenlijstOpvragenResponse response;
      int hoogsteKlantAfgifteNummer = -1;

      Gds2AfgifteServiceV20170401 gds2 = initGDS2();
      l.updateStatus("Uitvoeren SOAP request naar Kadaster...");

      boolean afgifteNummerGebaseerdOphalen = false;
      if (vanaf == null && tot == null) {
        //  instellen laagste en hoogste afgiftenummer voor dit verzoek, mag alleen als er
        // geen periode is gedefinieerd
        KlantAfgiftenummerReeksType afgiftenummers = new KlantAfgiftenummerReeksType();
        String nr =
            ClobElement.nullSafeGet(this.config.getConfig().get("klantafgiftenummer_vanaf"));
        if (StringUtils.isNotBlank(nr)) {
          BigInteger afgifteNrVanaf = new BigInteger(nr);
          afgiftenummers.setKlantAfgiftenummerVanaf(afgifteNrVanaf);
          log.info("Ophalen vanaf klant afgifte nummer: " + afgifteNrVanaf);
          l.addLog("Ophalen vanaf klant afgifte nummer: " + afgifteNrVanaf);

          nr = ClobElement.nullSafeGet(this.config.getConfig().get("klantafgiftenummer_totenmet"));
          if (StringUtils.isNotBlank(nr)) {
            BigInteger afgifteNrTM = new BigInteger(nr);
            afgiftenummers.setKlantAfgiftenummerTotmet(afgifteNrTM);
            log.info("Ophalen tot en met klant afgifte nummer: " + afgifteNrTM);
            l.addLog("Ophalen tot en met klant afgifte nummer: " + afgifteNrTM);
          }
          criteria.setKlantAfgiftenummerReeks(afgiftenummers);
          afgifteNummerGebaseerdOphalen = true;
        }
      }

      if (afgifteNummerGebaseerdOphalen) {
        // <editor-fold> afgiftenummer-based ophalen
        l.addLog(
            "GDS2 verzoek voor afgiftenummers vanaf: "
                + criteria.getKlantAfgiftenummerReeks().getKlantAfgiftenummerVanaf());
        l.addLog(
            "GDS2 verzoek voor afgiftenummers tot-en-met: "
                + criteria.getKlantAfgiftenummerReeks().getKlantAfgiftenummerTotmet());
        response =
            retryBestandenLijstOpvragen(
                gds2, request, BESTANDENLIJST_ATTEMPTS, BESTANDENLIJST_RETRY_WAIT);
        hoogsteKlantAfgifteNummer = response.getAntwoord().getKlantAfgiftenummerMax();
        boolean hasMore = response.getAntwoord().isMeerAfgiftesbeschikbaar();
        int aantalInAntwoord = response.getAntwoord().getAfgifteAantalInLijst();
        if (aantalInAntwoord > 0) {
          afgiftes.addAll(response.getAntwoord().getBestandenLijst().getAfgifte());
        }
        l.addLog("Aantal in antwoord: " + aantalInAntwoord);
        l.addLog("Meer afgiftes beschikbaar: " + hasMore);

        while (hasMore) {
          criteria
              .getKlantAfgiftenummerReeks()
              .setKlantAfgiftenummerVanaf(
                  // vanaf ophogen met aantal opgehaald
                  criteria
                      .getKlantAfgiftenummerReeks()
                      .getKlantAfgiftenummerVanaf()
                      .add(BigInteger.valueOf(aantalInAntwoord)));
          l.addLog(
              "GDS2 verzoek voor afgiftenummers vanaf: "
                  + criteria.getKlantAfgiftenummerReeks().getKlantAfgiftenummerVanaf());
          l.addLog(
              "GDS2 verzoek voor afgiftenummers tot-en-met: "
                  + criteria.getKlantAfgiftenummerReeks().getKlantAfgiftenummerTotmet());
          response =
              retryBestandenLijstOpvragen(
                  gds2, request, BESTANDENLIJST_ATTEMPTS, BESTANDENLIJST_RETRY_WAIT);
          hoogsteKlantAfgifteNummer = response.getAntwoord().getKlantAfgiftenummerMax();
          aantalInAntwoord = response.getAntwoord().getAfgifteAantalInLijst();
          hasMore = response.getAntwoord().isMeerAfgiftesbeschikbaar();
          l.addLog("Aantal in antwoord: " + aantalInAntwoord);
          l.addLog("Meer afgiftes beschikbaar: " + hasMore);

          if (aantalInAntwoord > 0) {
            afgiftes.addAll(response.getAntwoord().getBestandenLijst().getAfgifte());
          }
        }
        // </editor-fold> afgiftenummer-based ophalen
      } else {
        // <editor-fold> time-based ophalen
        GregorianCalendar currentMoment = null;
        boolean parseblePeriod = false;
        int loopType = Calendar.DAY_OF_MONTH;
        int loopMax = 180;
        int loopNum = 0;
        boolean reducePeriod = false;
        boolean increasePeriod = false;

        if (vanaf != null && tot != null && vanaf.before(tot)) {
          parseblePeriod = true;
          currentMoment = vanaf;
        }

        boolean morePeriods2Process = false;
        do /* while morePeriods2Process is true */ {
          l.addLog("\n*** start periode ***");
          // zet periode in criteria indien gewenst
          if (parseblePeriod) {
            // check of de periodeduur verkleind moet worden
            if (reducePeriod) {
              switch (loopType) {
                case Calendar.DAY_OF_MONTH:
                  currentMoment.add(loopType, -1);
                  loopType = Calendar.HOUR_OF_DAY;
                  l.addLog("* Verklein loop periode naar uur");
                  break;
                case Calendar.HOUR_OF_DAY:
                  currentMoment.add(loopType, -1);
                  loopType = Calendar.MINUTE;
                  l.addLog("* Verklein loop periode naar minuut");
                  break;
                case Calendar.MINUTE:
                default:
                  /*
                   * Hier kom je alleen als binnen een minuut meer
                   * dan 2000 berichten zijn aangamaakt en het
                   * vinkje ook "al rapporteerde berichten
                   * ophalen" staat aan.
                   */
                  l.addLog("Niet alle gevraagde berichten zijn opgehaald");
              }
              reducePeriod = false;
            }

            // check of de periodeduur vergroot moet worden
            if (increasePeriod) {
              switch (loopType) {
                case Calendar.HOUR_OF_DAY:
                  loopType = Calendar.DAY_OF_MONTH;
                  l.addLog("* Vergroot loop periode naar dag");
                  break;
                case Calendar.MINUTE:
                  loopType = Calendar.HOUR_OF_DAY;
                  l.addLog("* Vergroot loop periode naar uur");
                  break;
                case Calendar.DAY_OF_MONTH:
                default:
                  // not possible
              }
              increasePeriod = false;
            }

            FilterDatumTijdType d = new FilterDatumTijdType();
            d.setDatumTijdVanaf(getXMLDatumTijd(currentMoment));
            l.addLog(String.format("Datum vanaf: %tc", currentMoment.getTime()));
            currentMoment.add(loopType, 1);
            d.setDatumTijdTotmet(getXMLDatumTijd(currentMoment));
            l.addLog(String.format("Datum tot: %tc", currentMoment.getTime()));
            criteria.setPeriode(d);

            switch (loopType) {
              case Calendar.HOUR_OF_DAY:
                // 0-23
                if (currentMoment.get(loopType) == 0) {
                  increasePeriod = true;
                }
                break;
              case Calendar.MINUTE:
                // 0-59
                if (currentMoment.get(loopType) == 0) {
                  increasePeriod = true;
                }
                break;
              case Calendar.DAY_OF_MONTH:
              default:
                // alleen dagen tellen, uur en minuut altijd helemaal
                loopNum++;
            }

            // bereken of einde van periode bereikt is
            morePeriods2Process = currentMoment.before(tot) && loopNum < loopMax;
          }

          verzoek.setAfgifteSelectieCriteria(criteria);
          response =
              retryBestandenLijstOpvragen(
                  gds2, request, BESTANDENLIJST_ATTEMPTS, BESTANDENLIJST_RETRY_WAIT);
          hoogsteKlantAfgifteNummer = response.getAntwoord().getKlantAfgiftenummerMax();

          int aantalInAntwoord = response.getAntwoord().getAfgifteAantalInLijst();
          l.addLog("Aantal in antwoord: " + aantalInAntwoord);
          // opletten; in de xsd staat een default value van 'J' voor
          // meerAfgiftesbeschikbaar
          boolean hasMore = response.getAntwoord().isMeerAfgiftesbeschikbaar();
          l.addLog("Meer afgiftes beschikbaar: " + hasMore);

          /*
           * Als "al gerapporteerde berichten" moeten worden opgehaald en
           * er zitten dan 2000 berichten in het antwoord dan heeft het
           * geen zin om meer keer de berichten op te halen, je krijgt
           * telkens dezelfde.
           */
          if (hasMore && "true".equals(alGerapporteerd)) {
            reducePeriod = true;
            morePeriods2Process = true;
            increasePeriod = false;
            // als er geen parsable periode is
            // (geen periode ingevuld en alGerapporteerd is true
            // dan moet morePeriods2Process false worden om een
            // eindloze while loop te voorkomen
            if (!parseblePeriod) {
              morePeriods2Process = false;
            } else {
              continue;
            }
          }

          if (aantalInAntwoord > 0) {
            afgiftes.addAll(response.getAntwoord().getBestandenLijst().getAfgifte());
          }

          /*
           * Indicatie nog niet gerapporteerd: Met deze indicatie wordt
           * aangegeven of uitsluitend de nog niet gerapporteerde
           * bestanden moeten worden opgenomen in de lijst, of dat alle
           * beschikbare bestanden worden genoemd. Niet gerapporteerd
           * betekent in dit geval ‘niet eerder opgevraagd in deze
           * bestandenlijst’. Als deze indicator wordt gebruikt, dan
           * worden na terugmelding van de bestandenlijst de bijbehorende
           * bestanden gemarkeerd als zijnde ‘gerapporteerd’ in het
           * systeem van GDS.
           */
          int moreCount = 0;
          String dontGetMoreConfig =
              ClobElement.nullSafeGet(
                  this.config.getConfig().get("gds2_niet_gerapporteerde_afgiftes_niet_ophalen"));

          while (hasMore && !"true".equals(dontGetMoreConfig)) {
            l.updateStatus(
                "Uitvoeren SOAP request naar Kadaster voor meer afgiftes..." + moreCount++);
            criteria.setNogNietGerapporteerd(true);
            response =
                retryBestandenLijstOpvragen(
                    gds2, request, BESTANDENLIJST_ATTEMPTS, BESTANDENLIJST_RETRY_WAIT);
            hoogsteKlantAfgifteNummer = response.getAntwoord().getKlantAfgiftenummerMax();

            List<AfgifteType> afgifteLijst =
                response.getAntwoord().getBestandenLijst().getAfgifte();
            for (AfgifteType t : afgiftes) {
              // lijst urls naar aparte logfile loggen
              if (t.getDigikoppelingExternalDatareferences() != null
                  && t.getDigikoppelingExternalDatareferences().getDataReference() != null) {
                for (DataReference dr :
                    t.getDigikoppelingExternalDatareferences().getDataReference()) {
                  log.info(
                      "GDS2url te downloaden: "
                          + dr.getTransport().getLocation().getSenderUrl().getValue());
                  break;
                }
              }
            }
            afgiftes.addAll(afgifteLijst);
            aantalInAntwoord = response.getAntwoord().getAfgifteAantalInLijst();
            l.addLog("Aantal in antwoord: " + aantalInAntwoord);
            hasMore = response.getAntwoord().isMeerAfgiftesbeschikbaar();
            l.addLog("Nog meer afgiftes beschikbaar: " + hasMore);
          }

        } while (morePeriods2Process);
        // </editor-fold> time-based ophalen
      }
      l.total(afgiftes.size());
      l.addLog("Totaal aantal op te halen berichten: " + afgiftes.size());
      l.addLog("Hoogste klant afgifte nummer: " + hoogsteKlantAfgifteNummer);

      this.config
          .getConfig()
          .put("hoogste_afgiftenummer", new ClobElement("" + hoogsteKlantAfgifteNummer));

      final String soort =
          this.config.getConfig().getOrDefault("gds2_br_soort", new ClobElement("brk")).getValue();
      if (soort.equals("brk2")) {
        verwerkAfgiftes(afgiftes, getCertificaatBaseURL(response.getAntwoord()), soort);
      } else {
        throw new BrmoException("Onbekende basisregistratie soort: " + soort);
      }

    } catch (Exception e) {
      log.error("Fout bij ophalen van GDS2 berichten", e);
      this.config.setLastrun(new Date());
      this.config.setSamenvatting("Er is een fout opgetreden, details staan in de logs");
      this.config.setStatus(AutomatischProces.ProcessingStatus.ERROR);
      String m = "Fout bij ophalen van GDS2 berichten: " + ExceptionUtils.getMessage(e);
      if (e.getCause() != null) {
        m += ", oorzaak: " + ExceptionUtils.getRootCauseMessage(e);
      }
      l.addLog(m);
      l.exception(e);
    } finally {
      if (Stripersist.getEntityManager().getTransaction().getRollbackOnly()) {
        // XXX bij rollback only wordt status niet naar ERROR gezet vanwege
        // rollback, zou in aparte transactie moeten
        Stripersist.getEntityManager().getTransaction().rollback();
      } else {
        Stripersist.getEntityManager().merge(this.config);
        Stripersist.getEntityManager().getTransaction().commit();
      }
      l.updateStatus("Uitvoeren SOAP request naar Kadaster afgerond");
    }
  }

  private String getLaadprocesBestandsnaam(AfgifteType a) {
    return a.getBestand().getBestandsnaam() + " (" + a.getAfgifteID() + ")";
  }

  private boolean isAfgifteAlGeladen(AfgifteType a) {
    try {
      Stripersist.getEntityManager()
          .createQuery("select 1 from LaadProces lp where lp.bestand_naam = :n")
          .setParameter("n", getLaadprocesBestandsnaam(a))
          .getSingleResult();
      return true;
    } catch (NoResultException nre) {
      return false;
    }
  }

  private Bericht laadAfgifte(AfgifteType a, String url, String brkSoort) throws Exception {
    EntityManager em = Stripersist.getEntityManager();
    String msg = "Downloaden " + url;
    l.updateStatus(msg);
    l.addLog(msg);

    ByteArrayOutputStream bos = new ByteArrayOutputStream();

    int attempt = 0;
    while (true) {
      try {
        URLConnection connection = new URL(url).openConnection();
        if (connection instanceof HttpsURLConnection) {
          ((HttpsURLConnection) connection).setSSLSocketFactory(context.getSocketFactory());
        }
        InputStream input = (InputStream) connection.getContent();
        IOUtils.copy(input, bos);
        break;
      } catch (Exception e) {
        attempt++;
        if (attempt == AFGIFTE_DOWNLOAD_ATTEMPTS) {
          l.addLog(
              "Fout bij laatste poging downloaden afgifte: "
                  + e.getClass().getName()
                  + ": "
                  + e.getMessage());
          throw e;
        } else {
          l.addLog(
              "Fout bij poging "
                  + attempt
                  + " om afgifte te downloaden: "
                  + e.getClass().getName()
                  + ": "
                  + e.getMessage());
          Thread.sleep(AFGIFTE_DOWNLOAD_RETRY_WAIT);
          l.addLog(
              "Uitvoeren poging " + (attempt + 1) + " om afgifte " + url + " te downloaden...");
        }
      }
    }

    LaadProces lp = new LaadProces();
    lp.setBestand_naam(getLaadprocesBestandsnaam(a));
    lp.setKlantafgiftenummer(a.getKlantAfgiftenummer());
    lp.setContractafgiftenummer(a.getContractAfgiftenummer());
    lp.setArtikelnummer(a.getBestandKenmerken().getArtikelnummer());
    lp.setContractnummer(a.getBestandKenmerken().getContractnummer());
    lp.setAfgifteid(a.getAfgifteID());
    lp.setAfgiftereferentie(a.getAfgiftereferentie());
    lp.setBestandsreferentie(a.getBestand().getBestandsreferentie());
    lp.setBeschikbaar_tot(a.getBeschikbaarTot().toGregorianCalendar().getTime());
    lp.setBestand_naam_hersteld(a.getBestand().getBestandsnaam());

    if (a.getDigikoppelingExternalDatareferences() != null
        && a.getDigikoppelingExternalDatareferences().getDataReference() != null) {
      for (DataReference dr : a.getDigikoppelingExternalDatareferences().getDataReference()) {
        lp.setBestand_datum(
            dr.getLifetime().getCreationTime().getValue().toGregorianCalendar().getTime());
        break;
      }
    }
    lp.setSoort(brkSoort);
    lp.setStatus(LaadProces.STATUS.STAGING_OK);
    lp.setStatus_datum(new Date());
    SimpleDateFormat sdf = new SimpleDateFormat("dd-MM-yyyy HH:mm:ss");
    lp.setOpmerking("GDS2 download van " + url + " op " + sdf.format(new Date()));
    lp.setAutomatischProces(em.find(AutomatischProces.class, config.getId()));

    Bericht b = new Bericht();
    b.setLaadprocesid(lp);
    b.setDatum(lp.getBestand_datum());
    b.setSoort(brkSoort);
    b.setStatus_datum(new Date());

    ZipInputStream zip = new ZipInputStream(new ByteArrayInputStream(bos.toByteArray()));
    ZipEntry entry = zip.getNextEntry();
    while (entry != null && !entry.getName().toLowerCase().endsWith(".xml")) {
      msg = "Overslaan zip entry geen XML: " + entry.getName();
      l.addLog(msg);
      entry = zip.getNextEntry();
    }
    if (entry == null) {
      msg = "Geen geschikt XML bestand gevonden in zip bestand!";
      l.addLog(msg);
      return null;
    }
    b.setBr_orgineel_xml(IOUtils.toString(zip, StandardCharsets.UTF_8));

    if (log.isDebugEnabled()) {
      // dump xml bestand naar /tmp/ voordat de xml wordt geparsed
      String fName = File.createTempFile(lp.getBestand_naam(), ".xml").getAbsolutePath();
      log.debug("Dump xml bericht naar: " + fName);
      IOUtils.write(b.getBr_orgineel_xml(), new FileWriter(fName, StandardCharsets.UTF_8));
    }
    try {
      String brXML;
      Date brDatum;
      Integer brVolgordeNummer;
      String brObjectRef;
      Brk2SnapshotXMLReader reader2 =
          new Brk2SnapshotXMLReader(
              new ByteArrayInputStream(b.getBr_orgineel_xml().getBytes(StandardCharsets.UTF_8)));
      Brk2Bericht parsedBericht = reader2.next();
      brXML = parsedBericht.getBrXml();
      brDatum = parsedBericht.getDatum();
      brVolgordeNummer = parsedBericht.getVolgordeNummer();
      brObjectRef = parsedBericht.getObjectRef();

      if (null != brDatum) {
        b.setDatum(brDatum);
      }
      b.setBr_xml(brXML);
      b.setVolgordenummer(brVolgordeNummer);

      // Als objectRef niet opgehaald kan worden,dan kan het
      // bericht niet verwerkt worden.
      if (null != brObjectRef && !brObjectRef.isBlank()) {
        b.setObject_ref(brObjectRef);
        b.setStatus(Bericht.STATUS.STAGING_OK);
        b.setOpmerking("Klaar voor verwerking.");
      } else {
        b.setStatus(Bericht.STATUS.STAGING_NOK);
        b.setOpmerking("Object Ref niet gevonden in bericht-xml, neem contact op met leverancier.");
      }
    } catch (Exception e) {
      b.setStatus(Bericht.STATUS.STAGING_NOK);
      StringWriter sw = new StringWriter();
      e.printStackTrace(new PrintWriter(sw));
      b.setOpmerking("Fout bij parsen BRK bericht: " + sw);
    }
    try {
      em.persist(lp);
      em.persist(b);
      em.merge(this.config);
      em.flush();
      em.getTransaction().commit();
      em.clear();
    } catch (PersistenceException pe) {
      if (!em.getTransaction().isActive()) {
        em.getTransaction().begin();
      }
      em.getTransaction().rollback();
      em.getTransaction().begin();
      lp.setId(null);
      log.warn("Opslaan van bericht uit laadproces " + lp.getBestand_naam() + " is mislukt.", pe);
      log.warn(
          "Duplicaat bericht: "
              + b.getObject_ref()
              + ":"
              + b.getBr_orgineel_xml()
              + "("
              + b.getBr_xml()
              + ")");
      lp.setOpmerking(
          lp.getOpmerking() + ": Fout, duplicaat bericht. Berichtinhoud: \n\n" + b.getBr_xml());
      b = null;
      lp.setStatus(LaadProces.STATUS.STAGING_DUPLICAAT);
      lp.setAutomatischProces(em.find(AutomatischProces.class, this.config.getId()));
      em.merge(this.config);
      em.persist(lp);
      em.flush();
      em.getTransaction().commit();
      em.clear();
    }
    return b;
  }

  /**
   * Post de xml naar de geconfigureerde url.
   *
   * @param proces proces waarvoor doorgetuurd wordt
   * @param l update listener
   * @param b door te sturen bericht
   * @param endpoint url waarnaartoe wordt gepost
   * @return {@code true} als succesvol doorgestuurd
   */
  public static boolean doorsturenBericht(
      AutomatischProces proces, ProgressUpdateListener l, Bericht b, String endpoint) {
    String msg;
    SimpleDateFormat sdf = new SimpleDateFormat("dd-MM-yyyy HH:mm:ss");
    try {
      if (endpoint == null || endpoint.length() < 1) {
        return false;
      }
      URL url = new URL(endpoint);
      HttpURLConnection conn = (HttpURLConnection) url.openConnection();
      conn.setDoOutput(true);
      conn.setRequestMethod("POST");
      conn.setRequestProperty("Content-Length", "" + b.getBr_orgineel_xml().length());
      conn.setRequestProperty("Content-Type", "application/octet-stream");
      IOUtils.copy(
          IOUtils.toInputStream(b.getBr_orgineel_xml(), StandardCharsets.UTF_8),
          conn.getOutputStream());
      conn.disconnect();
      if ((conn.getResponseCode() == HttpURLConnection.HTTP_OK)
          || (conn.getResponseCode() == HttpURLConnection.HTTP_CREATED)
          || (conn.getResponseCode() == HttpURLConnection.HTTP_ACCEPTED)) {
        msg =
            "Bericht "
                + b.getObject_ref()
                + " op "
                + sdf.format(new Date())
                + " doorgestuurd naar "
                + endpoint;
        b.setStatus(Bericht.STATUS.STAGING_FORWARDED);
      } else {
        msg =
            String.format(
                "HTTP foutcode bij doorsturen bericht %s op %s naar endpoint %s: %s",
                b.getObject_ref(),
                sdf.format(new Date()),
                endpoint,
                conn.getResponseCode() + ": " + conn.getResponseMessage());
        b.setStatus(Bericht.STATUS.STAGING_NOK);
      }
      b.setOpmerking(msg);
      l.addLog(msg);

      return b.getStatus() == Bericht.STATUS.STAGING_FORWARDED;
    } catch (Exception e) {
      msg =
          String.format(
              "Fout bij doorsturen bericht op %s naar endpoint %s: %s",
              sdf.format(new Date()), endpoint, e.getClass() + ": " + e.getMessage());
      b.setOpmerking(msg);
      b.setStatus(Bericht.STATUS.STAGING_NOK);
      l.addLog(msg);
      log.error(msg, e);
      return false;
    }
  }

  /**
   * verwerk de afgiftes in de response en stuur eventueel door.
   *
   * @param afgiftes lijst afgiftes
   * @throws Exception if any
   */
  private void verwerkAfgiftes(List<AfgifteType> afgiftes, BaseURLType baseUrl, final String soort)
      throws Exception {
    int filterAlVerwerkt = 0;
    int aantalGeladen = 0;
    int aantalDoorgestuurd = 0;
    int progress = 0;
    List<Long> geladenBerichtIds = new ArrayList<>();
    String doorsturenUrl =
        ClobElement.nullSafeGet(this.config.getConfig().get("delivery_endpoint"));

    for (AfgifteType a : afgiftes) {
      String url = getAfgifteURL(a, baseUrl);

      if (null != url && !url.isBlank()) {
        if (isAfgifteAlGeladen(a)) {
          filterAlVerwerkt++;
        } else {
          Bericht b = laadAfgifte(a, url, soort);
          if (b != null && doorsturenUrl != null) {
            geladenBerichtIds.add(b.getId());
          }
          aantalGeladen++;
        }
      }
      l.progress(++progress);
    }

    Bericht b;
    if (doorsturenUrl != null && !geladenBerichtIds.isEmpty()) {
      l.addLog("Doorsturen van de opgehaalde berichten, in totaal " + geladenBerichtIds.size());
      l.total(geladenBerichtIds.size());
      progress = 0;

      for (Long bId : geladenBerichtIds) {
        b = Stripersist.getEntityManager().find(Bericht.class, bId);
        l.updateStatus("Doorsturen bericht: " + b.getObject_ref());
        doorsturenBericht(this.config, l, b, doorsturenUrl);
        Stripersist.getEntityManager().merge(b);
        Stripersist.getEntityManager().merge(this.config);
        Stripersist.getEntityManager().flush();
        Stripersist.getEntityManager().getTransaction().commit();
        Stripersist.getEntityManager().clear();
        aantalDoorgestuurd++;
        l.progress(++progress);
      }
    }

    l.addLog("\n\n**** resultaat ****\n");
    l.addLog("Aantal afgiftes die al waren verwerkt: " + filterAlVerwerkt);
    l.addLog("\nAantal afgiftes geladen: " + aantalGeladen);
    l.addLog("\nAantal afgiftes doorgestuurd: " + aantalDoorgestuurd + "\n");

    l.addLog(
        String.format(
            "Het GDS2 ophalen proces met ID %d is afgerond op %tc",
            config.getId(), Calendar.getInstance()));
    this.config.updateSamenvattingEnLogfile(
        String.format(
            "Het GDS2 ophalen proces, gestart op %tc, is afgerond op %tc. "
                + LOG_NEWLINE
                + "Aantal afgiftes die al waren verwerkt: %d"
                + LOG_NEWLINE
                + "Aantal afgiftes geladen: %d"
                + LOG_NEWLINE
                + "Aantal afgiftes doorgestuurd: %d"
                + LOG_NEWLINE
                + "Hoogste klantafgiftenummer: %s",
            this.config.getLastrun(),
            Calendar.getInstance(),
            filterAlVerwerkt,
            aantalGeladen,
            aantalDoorgestuurd,
            ClobElement.nullSafeGet(this.config.getConfig().get("hoogste_afgiftenummer"))));

    this.config.setStatus(AutomatischProces.ProcessingStatus.WAITING);
    this.config.setLastrun(new Date());
    Stripersist.getEntityManager().merge(this.config);
    // om geheugen problemen te vookomen bij runs met veel downloads en berichten
    // een flush/commit/clear forceren
    Stripersist.getEntityManager().flush();
    Stripersist.getEntityManager().getTransaction().commit();
    Stripersist.getEntityManager().clear();
  }

  private Gds2AfgifteServiceV20170401 initGDS2() throws Exception {
    Gds2AfgifteServiceV20170401 gds2 =
        new Gds2AfgifteServiceV20170401Service().getAGds2AfgifteServiceV20170401();
    BindingProvider bp = (BindingProvider) gds2;
    Map<String, Object> ctxt = bp.getRequestContext();

    // soap berichten logger inhaken (actief met TRACE level)
    List<Handler> handlerChain = bp.getBinding().getHandlerChain();
    handlerChain.add(new LogMessageHandler());
    bp.getBinding().setHandlerChain(handlerChain);

    String endpoint = (String) ctxt.get(BindingProvider.ENDPOINT_ADDRESS_PROPERTY);
    l.addLog("Kadaster endpoint: " + endpoint);
    l.updateStatus("Laden keys...");

    l.addLog("Initializing KeyManagerFactory");
    KeyManagerFactory kmf = KeyManagerFactory.getInstance(KeyManagerFactory.getDefaultAlgorithm());
    KeyStore ks = KeyStore.getInstance("jks");
    final char[] thePassword = "changeit".toCharArray();
    PrivateKey privateKey =
        getPrivateKeyFromPEM(this.config.getConfig().get("gds2_privkey").getValue());
    Certificate certificate =
        getCertificateFromPEM(this.config.getConfig().get("gds2_pubkey").getValue());
    ks.load(null);
    ks.setKeyEntry("thekey", privateKey, thePassword, new Certificate[] {certificate});
    kmf.init(ks, thePassword);

    l.updateStatus("Opzetten SSL context...");
    context = createSslContext(kmf);

    Client client = ClientProxy.getClient(gds2);
    HTTPConduit http = (HTTPConduit) client.getConduit();
    TLSClientParameters tlsClientParameters = new TLSClientParameters();
    tlsClientParameters.setSSLSocketFactory(context.getSocketFactory());
    http.setTlsClientParameters(tlsClientParameters);

    return gds2;
  }

  private SSLContext createSslContext(KeyManagerFactory kmf)
      throws KeyStoreException,
          IOException,
          CertificateException,
          NoSuchProviderException,
          NoSuchAlgorithmException,
          KeyManagementException {
    final SSLContext sslContext = SSLContext.getInstance("TLS", "SunJSSE");

    l.addLog("Loading PKIX Overheid keystore");
    KeyStore ks = KeyStore.getInstance("jks");
    ks.load(
        nl.b3p.gds2.GDS2Util.class.getResourceAsStream("/pkioverheid.jks"),
        "changeit".toCharArray());

    l.addLog("Initializing default TrustManagerFactory");
    final TrustManagerFactory javaDefaultTrustManager =
        TrustManagerFactory.getInstance(TrustManagerFactory.getDefaultAlgorithm());
    javaDefaultTrustManager.init((KeyStore) null);
    X509TrustManager defaultX509TrustManager = null;
    for (TrustManager t : javaDefaultTrustManager.getTrustManagers()) {
      if (t instanceof X509TrustManager) {
        defaultX509TrustManager = (X509TrustManager) t;
        break;
      }
    }
    l.addLog("Initializing PKIX TrustManagerFactory");
    final TrustManagerFactory customCaTrustManager = TrustManagerFactory.getInstance("PKIX");
    customCaTrustManager.init(ks);

    l.addLog("Initializing SSLContext");
    sslContext.init(
        kmf.getKeyManagers(),
        new TrustManager[] {
          new TrustManagerDelegate(
              // customCaTrustManager is PKIX dus altijd X.509 (RFC3280)
              (X509TrustManager) customCaTrustManager.getTrustManagers()[0],
              defaultX509TrustManager)
        },
        null);
    return sslContext;
  }
}