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

se.swedenconnect.signservice.application.DefaultSignServiceEngineManager Maven / Gradle / Ivy

/*
 * Copyright 2022-2024 Sweden Connect
 *
 * 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 se.swedenconnect.signservice.application;

import java.util.Collections;
import java.util.List;
import java.util.Objects;

import jakarta.annotation.Nonnull;
import jakarta.annotation.Nullable;
import lombok.extern.slf4j.Slf4j;
import se.swedenconnect.signservice.audit.AuditEventIds;
import se.swedenconnect.signservice.audit.AuditLogger;
import se.swedenconnect.signservice.context.SignServiceContext;
import se.swedenconnect.signservice.core.http.HttpUserRequest;
import se.swedenconnect.signservice.engine.SignServiceEngine;
import se.swedenconnect.signservice.engine.UnrecoverableErrorCodes;
import se.swedenconnect.signservice.engine.UnrecoverableSignServiceException;

/**
 * The main application bean. The SignService engine manager is responsible of routing every call to a matching engine.
 */
@Slf4j
public class DefaultSignServiceEngineManager implements SignServiceEngineManager {

  /** The SignService engines. */
  private final List engines;

  /** The system audit logger. */
  private final AuditLogger systemAuditLogger;

  /**
   * Constructor accepting the list of SignService engines that are configured for the SignService application.
   *
   * @param engines a list of SignService engines
   * @param systemAuditLogger the system audit logger
   */
  public DefaultSignServiceEngineManager(
      @Nonnull final List engines, @Nonnull final AuditLogger systemAuditLogger) {
    this.engines = Objects.requireNonNull(engines, "engines must not be null");
    this.systemAuditLogger = Objects.requireNonNull(systemAuditLogger, "systemAuditLogger must not be null");

    this.systemAuditLogger.auditLog(AuditEventIds.EVENT_SYSTEM_STARTED, (b) -> b.build());
  }

  /** {@inheritDoc} */
  @Override
  @Nonnull
  public SignServiceProcessingResult processRequest(
      @Nonnull final HttpUserRequest request, @Nullable final SignServiceContext signServiceContext)
      throws UnrecoverableSignServiceException {

    log.debug("Received {} request [url: '{}', client-ip: '{}']",
        request.getMethod(), request.getRequestUrl(), request.getClientIpAddress());

    // Find an engine that can process the request ...
    //
    final SignServiceEngine engine = this.engines.stream()
        .filter(e -> e.canProcess(request))
        .findFirst()
        .orElse(null);

    if (engine == null) {
      log.info("No SignServiceEngine can service {} request on {}", request.getMethod(), request.getRequestUrl());
      this.systemAuditLogger.auditLog(AuditEventIds.EVENT_SYSTEM_NOTFOUND,
          (b) -> b
              .parameter("url", request.getRequestUrl())
              .parameter("method", request.getMethod())
              .build());

      throw new UnrecoverableSignServiceException(UnrecoverableErrorCodes.NOT_FOUND, "No such resource");
    }
    log.debug("Engine '{}' is processing {} request [url: '{}']", engine.getName(), request.getMethod(),
        request.getRequestUrl());

    // Hand the request over to the engine ...
    //
    try {
      final SignServiceProcessingResult result = engine.processRequest(request, signServiceContext);

      log.debug("Engine '{}' processed request {} '{}': {}", engine.getName(), request.getMethod(),
          request.getRequestUrl(), result);

      return result;
    }
    catch (final UnrecoverableSignServiceException e) {
      final String msg = String.format("Engine '%s' reported error '%s' when processing request received on '%s' - %s",
          engine.getName(), e.getErrorCode(), request.getRequestUrl(), e.getMessage());
      log.info("{}", msg, e);

      this.systemAuditLogger.auditLog(AuditEventIds.EVENT_SYSTEM_PROCESSING_ERROR, (b) -> b
          .parameter("url", request.getRequestUrl())
          .parameter("engine-name", engine.getName())
          .parameter("error-code", e.getErrorCode())
          .parameter("error-message", msg)
          .build());

      throw e;
    }
  }

  /** {@inheritDoc} */
  @Override
  @Nonnull
  public List getEngines() {
    return Collections.unmodifiableList(this.engines);
  }

  /** {@inheritDoc} */
  @Override
  @Nonnull
  public AuditLogger getSystemAuditLogger() {
    return this.systemAuditLogger;
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy