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

com.helger.phase4.bdew.Phase4BDEWSender Maven / Gradle / Ivy

There is a newer version: 3.0.1
Show newest version
/*
 * Copyright (C) 2023-2024 Gregor Scholtysik (www.soptim.de)
 * gregor[dot]scholtysik[at]soptim[dot]de
 *
 * Copyright (C) 2023-2024 Philip Helger (www.helger.com)
 * philip[at]helger[dot]com
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *         http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package com.helger.phase4.bdew;

import java.io.IOException;
import java.time.LocalDate;

import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import javax.annotation.OverridingMethodsMustInvokeSuper;
import javax.annotation.concurrent.Immutable;
import javax.annotation.concurrent.NotThreadSafe;

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

import com.helger.commons.string.ToStringGenerator;
import com.helger.phase4.attachment.AS4OutgoingAttachment;
import com.helger.phase4.attachment.WSS4JAttachment;
import com.helger.phase4.crypto.ECryptoAlgorithmC14N;
import com.helger.phase4.crypto.ECryptoKeyEncryptionAlgorithm;
import com.helger.phase4.crypto.ECryptoKeyIdentifierType;
import com.helger.phase4.profile.bdew.AS4BDEWProfileRegistarSPI;
import com.helger.phase4.profile.bdew.BDEWPMode;
import com.helger.phase4.sender.AbstractAS4UserMessageBuilderMIMEPayload;
import com.helger.phase4.util.AS4ResourceHelper;

/**
 * This class contains all the specifics to send AS4 messages with the BDEW
 * profile. See sendAS4Message as the main method to trigger the
 * sending, with all potential customization.
 *
 * @author Gregor Scholtysik
 * @author Philip Helger
 */
@Immutable
public final class Phase4BDEWSender
{
  private static final Logger LOGGER = LoggerFactory.getLogger (Phase4BDEWSender.class);

  private Phase4BDEWSender ()
  {}

  /**
   * @return Create a new Builder for AS4 messages if the payload is present.
   *         Never null.
   */
  @Nonnull
  public static BDEWUserMessageBuilder builder ()
  {
    return new BDEWUserMessageBuilder ();
  }

  /**
   * Abstract BDEW UserMessage builder class with sanity methods
   *
   * @author Philip Helger
   * @param 
   *        The implementation type
   */
  public abstract static class AbstractBDEWUserMessageBuilder >
                                                              extends
                                                              AbstractAS4UserMessageBuilderMIMEPayload 
  {
    // Default per section 2.2.6.2.1
    public static final ECryptoKeyIdentifierType DEFAULT_KEY_IDENTIFIER_TYPE_SIGN = ECryptoKeyIdentifierType.BST_DIRECT_REFERENCE;
    // Default per section 2.2.6.2.2
    public static final ECryptoKeyIdentifierType DEFAULT_KEY_IDENTIFIER_TYPE_CRYPT = ECryptoKeyIdentifierType.SKI_KEY_IDENTIFIER;

    private BDEWPayloadParams m_aPayloadParams;

    protected AbstractBDEWUserMessageBuilder ()
    {
      // Override default values
      try
      {
        as4ProfileID (AS4BDEWProfileRegistarSPI.AS4_PROFILE_ID);

        httpClientFactory (new Phase4BDEWHttpClientSettings ());

        // Other crypt parameters are located in the PMode security part
        cryptParams ().setKeyIdentifierType (DEFAULT_KEY_IDENTIFIER_TYPE_CRYPT);
        cryptParams ().setKeyEncAlgorithm (ECryptoKeyEncryptionAlgorithm.ECDH_ES_KEYWRAP_AES_128);
        cryptParams ().setEncryptSymmetricSessionKey (false);

        /**
         * Assumption: the BST "ValueType" attribute is set to
         * "http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-x509-token-profile-1.0#X509PKIPathv1"
         * by WSS4J automatically (see WSSecSignature#addBST)
         */

        // See BDEW specs 2.2.6.2
        // Other signing parameters are located in the PMode security part
        signingParams ().setKeyIdentifierType (DEFAULT_KEY_IDENTIFIER_TYPE_SIGN);
        signingParams ().setAlgorithmC14N (ECryptoAlgorithmC14N.C14N_EXCL_OMIT_COMMENTS);
        // Use the BST value type "#X509PKIPathv1"
        signingParams ().setUseSingleCertificate (false);

        // Must be empty
        conversationID ("");

        agreementRef (BDEWPMode.DEFAULT_AGREEMENT_ID);

        // According to #186
        forceMimeMessage (true);
      }
      catch (final Exception ex)
      {
        throw new IllegalStateException ("Failed to init AS4 Client builder", ex);
      }
    }

    /**
     * Encryption Key Identifier Type.
     *
     * @param eEncryptionKeyIdentifierType
     *        {@link ECryptoKeyIdentifierType}. Defaults to
     *        BST_DIRECT_REFERENCE. May be null.
     * @return this for chaining
     */
    @Nonnull
    public final IMPLTYPE encryptionKeyIdentifierType (@Nullable final ECryptoKeyIdentifierType eEncryptionKeyIdentifierType)
    {
      if (eEncryptionKeyIdentifierType != null)
        cryptParams ().setKeyIdentifierType (eEncryptionKeyIdentifierType);
      return thisAsT ();
    }

    /**
     * Signing Key Identifier Type.
     *
     * @param eSigningKeyIdentifierType
     *        {@link ECryptoKeyIdentifierType}. Defaults to
     *        BST_DIRECT_REFERENCE. May be null.
     * @return this for chaining
     * @deprecated Use
     *             {@link #signingKeyIdentifierType(ECryptoKeyIdentifierType)}
     *             instead
     */
    @Nonnull
    @Deprecated (since = "2.1.2", forRemoval = true)
    public final IMPLTYPE setSigningKeyIdentifierType (@Nullable final ECryptoKeyIdentifierType eSigningKeyIdentifierType)
    {
      return signingKeyIdentifierType (eSigningKeyIdentifierType);
    }

    /**
     * Signing Key Identifier Type.
     *
     * @param eSigningKeyIdentifierType
     *        {@link ECryptoKeyIdentifierType}. Defaults to
     *        BST_DIRECT_REFERENCE. May be null.
     * @return this for chaining
     */
    @Nonnull
    public final IMPLTYPE signingKeyIdentifierType (@Nullable final ECryptoKeyIdentifierType eSigningKeyIdentifierType)
    {
      if (eSigningKeyIdentifierType != null)
        signingParams ().setKeyIdentifierType (eSigningKeyIdentifierType);
      return thisAsT ();
    }

    /**
     * Set the payload to be send out.
     *
     * @param aBuilder
     *        The payload builder to be used. GZip compression is automatically.
     *        enforced.
     * @param aPayloadParams
     *        The payload params to use. May be null.
     * @return this for chaining
     */
    @Nonnull
    public final IMPLTYPE payload (@Nullable final AS4OutgoingAttachment.Builder aBuilder,
                                   @Nullable final BDEWPayloadParams aPayloadParams)
    {
      payload (aBuilder != null ? aBuilder.compressionGZIP ().build () : null);
      m_aPayloadParams = aPayloadParams;
      return thisAsT ();
    }

    @Override
    @OverridingMethodsMustInvokeSuper
    public boolean isEveryRequiredFieldSet ()
    {
      if (!super.isEveryRequiredFieldSet ())
        return false;

      if (m_aPayload == null)
      {
        // See #180 for the specifics
        if (m_sAction.equals (BDEWPMode.ACTION_REQUEST_SWITCH) || m_sAction.equals (BDEWPMode.ACTION_CONFIRM_SWITCH))
        {
          // Payload is optional
        }
        else
        {
          // Payload is mandatory
          LOGGER.warn ("The field 'payload' is not set");
          return false;
        }
      }

      if (!"".equals (m_sConversationID))
      {
        LOGGER.warn ("The field 'conversationID' must not be changed");
        return false;
      }

      // All valid
      return true;
    }

    @Override
    @Nullable
    protected WSS4JAttachment createMainAttachment (@Nonnull final AS4OutgoingAttachment aPayload,
                                                    @Nonnull final AS4ResourceHelper aResHelper) throws IOException
    {
      final WSS4JAttachment aPayloadAttachment = WSS4JAttachment.createOutgoingFileAttachment (aPayload, aResHelper);

      if (m_aPayloadParams != null)
      {
        if (m_aPayloadParams.getDocumentType () != null)
          aPayloadAttachment.customPartProperties ().put ("BDEWDocumentType", m_aPayloadParams.getDocumentType ());
        if (m_aPayloadParams.getDocumentDate () != null)
          aPayloadAttachment.customPartProperties ()
                            .put ("BDEWDocumentDate", m_aPayloadParams.getDocumentDate ().toString ());
        if (m_aPayloadParams.getDocumentNumber () != null)
          aPayloadAttachment.customPartProperties ().put ("BDEWDocumentNo", m_aPayloadParams.getDocumentNumber ());
        if (m_aPayloadParams.getFulfillmentDate () != null)
          aPayloadAttachment.customPartProperties ()
                            .put ("BDEWFulfillmentDate", m_aPayloadParams.getFulfillmentDate ().toString ());
        if (m_aPayloadParams.getSubjectPartyId () != null)
          aPayloadAttachment.customPartProperties ().put ("BDEWSubjectPartyID", m_aPayloadParams.getSubjectPartyId ());
        if (m_aPayloadParams.getSubjectPartyRole () != null)
          aPayloadAttachment.customPartProperties ()
                            .put ("BDEWSubjectPartyRole", m_aPayloadParams.getSubjectPartyRole ());
      }
      return aPayloadAttachment;
    }
  }

  /**
   * The builder class for sending AS4 messages using BDEW profile specifics.
   * Use {@link #sendMessage()} to trigger the main transmission.
   *
   * @author Philip Helger
   */
  public static class BDEWUserMessageBuilder extends AbstractBDEWUserMessageBuilder 
  {
    public BDEWUserMessageBuilder ()
    {}
  }

  /**
   * Additional parameters to add in the PayloadInfo part of AS4 UserMessage
   *
   * @author Gregor Scholtysik
   */
  @NotThreadSafe
  public static class BDEWPayloadParams
  {
    private String m_sDocumentType;
    private LocalDate m_aDocumentDate;
    private String m_sDocumentNumber;
    private LocalDate m_aFulfillmentDate;
    private String m_sSubjectPartyID;
    private String m_sSubjectPartyRole;

    /**
     * @return BDEW payload document type for payload identifier
     *         BDEWDocumentType
     */
    @Nullable
    public String getDocumentType ()
    {
      return m_sDocumentType;
    }

    /**
     * BDEW payload document type
     *
     * @param sDocumentType
     *        Document type to use. May be null.
     * @return this for chaining
     */
    @Nonnull
    public BDEWPayloadParams setDocumentType (@Nullable final String sDocumentType)
    {
      m_sDocumentType = sDocumentType;
      return this;
    }

    /**
     * @return BDEW payload document date for payload identifier
     *         BDEWDocumentDate
     */
    @Nullable
    public LocalDate getDocumentDate ()
    {
      return m_aDocumentDate;
    }

    /**
     * BDEW payload document date
     *
     * @param sDocumentDate
     *        Document date to use. May be null.
     * @return this for chaining
     */
    @Nonnull
    public BDEWPayloadParams setDocumentDate (@Nullable final LocalDate sDocumentDate)
    {
      m_aDocumentDate = sDocumentDate;
      return this;
    }

    /**
     * Note: type change in 2.1.3 from Integer to
     * String
     *
     * @return BDEW payload document number for payload identifier
     *         BDEWDocumentNo
     */
    @Nullable
    public String getDocumentNumber ()
    {
      return m_sDocumentNumber;
    }

    /**
     * BDEW payload document number
     *
     * @param nDocumentNumber
     *        Document number to use. May be null.
     * @return this for chaining
     * @since 2.1.2
     */
    @Nonnull
    public BDEWPayloadParams setDocumentNumber (final int nDocumentNumber)
    {
      return setDocumentNumber (Integer.toString (nDocumentNumber));
    }

    /**
     * BDEW payload document number.
* Note: type change in 2.1.3 from Integer to * String * * @param sDocumentNumber * Document number to use. May be null. * @return this for chaining */ @Nonnull public BDEWPayloadParams setDocumentNumber (@Nullable final String sDocumentNumber) { m_sDocumentNumber = sDocumentNumber; return this; } /** * @return BDEW payload fulfillment date for payload identifier * BDEWFulfillmentDate */ @Nullable public LocalDate getFulfillmentDate () { return m_aFulfillmentDate; } /** * BDEW payload fulfillment date * * @param sFulfillmenttDate * Fulfillment date to use. May be null. * @return this for chaining */ @Nonnull public BDEWPayloadParams setFulfillmentDate (@Nullable final LocalDate sFulfillmenttDate) { m_aFulfillmentDate = sFulfillmenttDate; return this; } /** * @return BDEW payload subject party ID for payload identifier * BDEWSubjectPartyID */ @Nullable public String getSubjectPartyId () { return m_sSubjectPartyID; } /** * BDEW payload subject party ID * * @param sSubjectPartyId * Subject party ID to use. May be null. * @return this for chaining */ @Nonnull public BDEWPayloadParams setSubjectPartyId (@Nullable final String sSubjectPartyId) { m_sSubjectPartyID = sSubjectPartyId; return this; } /** * @return BDEW payload subject party ID for payload identifier * BDEWSubjectPartyRole */ @Nullable public String getSubjectPartyRole () { return m_sSubjectPartyRole; } /** * BDEW payload subject party role * * @param sSubjectPartyRole * Subject party role to use. May be null. * @return this for chaining */ @Nonnull public BDEWPayloadParams setSubjectPartyRole (@Nullable final String sSubjectPartyRole) { m_sSubjectPartyRole = sSubjectPartyRole; return this; } @Override public String toString () { return new ToStringGenerator (this).append ("DocumentType", m_sDocumentType) .append ("DocumentDate", m_aDocumentDate) .append ("DocumentNumber", m_sDocumentNumber) .append ("FulfillmentDate", m_aFulfillmentDate) .append ("SubjectPartyID", m_sSubjectPartyID) .append ("SubjectPartyRole", m_sSubjectPartyRole) .getToString (); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy