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

io.fabric8.kubernetes.client.server.mock.KubernetesCrudDispatcher Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (C) 2015 Red Hat, Inc.
 *
 * 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 io.fabric8.kubernetes.client.server.mock;

import io.fabric8.kubernetes.api.model.GenericKubernetesResource;
import io.fabric8.kubernetes.client.Watcher.Action;
import io.fabric8.kubernetes.client.dsl.base.CustomResourceDefinitionContext;
import io.fabric8.kubernetes.client.server.mock.crud.KubernetesCrudDispatcherException;
import io.fabric8.kubernetes.client.server.mock.crud.KubernetesCrudDispatcherHandler;
import io.fabric8.kubernetes.client.server.mock.crud.KubernetesCrudPersistence;
import io.fabric8.kubernetes.client.server.mock.crud.PatchHandler;
import io.fabric8.kubernetes.client.server.mock.crud.PostHandler;
import io.fabric8.kubernetes.client.server.mock.crud.PutHandler;
import io.fabric8.kubernetes.client.utils.Serialization;
import io.fabric8.kubernetes.client.utils.Utils;
import io.fabric8.mockwebserver.Context;
import io.fabric8.mockwebserver.crud.Attribute;
import io.fabric8.mockwebserver.crud.AttributeSet;
import io.fabric8.mockwebserver.crud.CrudDispatcher;
import io.fabric8.mockwebserver.http.MockResponse;
import io.fabric8.mockwebserver.http.RecordedRequest;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.net.HttpURLConnection;
import java.net.URI;
import java.net.URISyntaxException;
import java.time.ZoneOffset;
import java.time.ZonedDateTime;
import java.time.format.DateTimeFormatter;
import java.time.temporal.ChronoUnit;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import java.util.concurrent.CopyOnWriteArraySet;
import java.util.concurrent.atomic.AtomicLong;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock;
import java.util.function.Supplier;

public class KubernetesCrudDispatcher extends CrudDispatcher implements KubernetesCrudPersistence, CustomResourceAware {

  private static final Logger LOGGER = LoggerFactory.getLogger(KubernetesCrudDispatcher.class);
  private final Set watchEventListeners;
  private final CustomResourceDefinitionProcessor crdProcessor;
  private final KubernetesAttributesExtractor kubernetesAttributesExtractor;
  private final AtomicLong resourceVersion;
  private final KubernetesResponseComposer kubernetesResponseComposer;
  private final KubernetesCrudDispatcherHandler postHandler;
  private final KubernetesCrudDispatcherHandler putHandler;
  private final KubernetesCrudDispatcherHandler patchHandler;
  private final ReadWriteLock lock = new ReentrantReadWriteLock();

  public KubernetesCrudDispatcher() {
    this(Collections.emptyList());
  }

  public KubernetesCrudDispatcher(List crdContexts) {
    super(new Context(Serialization.jsonMapper()), new KubernetesAttributesExtractor(), new KubernetesResponseComposer());
    this.kubernetesAttributesExtractor = (KubernetesAttributesExtractor) this.attributeExtractor;
    this.kubernetesResponseComposer = (KubernetesResponseComposer) this.responseComposer;
    watchEventListeners = new CopyOnWriteArraySet<>();
    crdProcessor = new CustomResourceDefinitionProcessor();
    this.kubernetesAttributesExtractor.setCustomResourceDefinitionProcessor(crdProcessor);
    resourceVersion = new AtomicLong();

    postHandler = new PostHandler(this.kubernetesAttributesExtractor, this);
    putHandler = new PutHandler(this);
    patchHandler = new PatchHandler(this);
    crdContexts.forEach(this::expectCustomResource);
  }

  MockResponse process(RecordedRequest request, KubernetesCrudDispatcherHandler handler) {
    return withLock(lock.writeLock(), () -> {
      try {
        return handler.handle(request);
      } catch (KubernetesCrudDispatcherException e) {
        return new MockResponse().setResponseCode(e.getCode()).setBody(e.toStatusBody());
      }
    });
  }

  /**
   * Adds the specified object to the in-memory db.
   *
   * @param request RecordedRequest information.
   * @return a MockResponse to be dispatched.
   */
  @Override
  public MockResponse handleCreate(RecordedRequest request) {
    return process(request, postHandler);
  }

  /**
   * Updates the specified object to the in-memory db.
   *
   * @param request RecordedRequest information.
   * @return a MockResponse to be dispatched.
   */
  @Override
  public MockResponse handleUpdate(RecordedRequest request) {
    return process(request, putHandler);
  }

  /**
   * Performs a get for the corresponding object from the in-memory db.
   *
   * @param path The path.
   * @return The {@link MockResponse}
   */
  @Override
  public MockResponse handleGet(String path) {
    return withLock(lock.readLock(), () -> {
      if (detectWatchMode(path)) {
        return handleWatch(path);
      }
      return handle(path, null);
    });
  }

  private interface EventProcessor {
    void processEvent(String path, AttributeSet pathAttributes, AttributeSet oldAttributes);
  }

  private MockResponse handle(String path, EventProcessor eventProcessor) {
    MockResponse response = new MockResponse();
    List items = new ArrayList<>();
    AttributeSet query = attributeExtractor.fromPath(path);

    new ArrayList<>(map.entrySet()).stream()
        .filter(entry -> entry.getKey().matches(query))
        .forEach(entry -> {
          LOGGER.debug("Entry found for query {} : {}", query, entry);
          items.add(entry.getValue());
          if (eventProcessor != null) {
            eventProcessor.processEvent(path, query, entry.getKey());
          }
        });

    if (query.containsKey(KubernetesAttributesExtractor.NAME)) {
      if (!items.isEmpty()) {
        response.setBody(items.get(0));
        response.setResponseCode(HttpURLConnection.HTTP_OK);
      } else {
        response.setResponseCode(HttpURLConnection.HTTP_NOT_FOUND);
      }
    } else {
      // if there are no items, and this is a get it may be an api metadata request
      String metadataResult = null;
      if (items.isEmpty() && eventProcessor == null) {
        metadataResult = crdProcessor.getApiResources(path);
      }
      if (metadataResult != null) {
        response.setBody(metadataResult);
      } else {
        response.setBody(kubernetesResponseComposer.compose(items, String.valueOf(resourceVersion.get())));
      }
      response.setResponseCode(HttpURLConnection.HTTP_OK);
    }
    return response;
  }

  /**
   * Patches the specified object to the in-memory db.
   *
   * @return The {@link MockResponse}
   */
  @Override
  public MockResponse handlePatch(RecordedRequest request) {
    return process(request, patchHandler);
  }

  /**
   * Performs a delete for the corresponding object from the in-memory db.
   *
   * @param path String
   * @return The {@link MockResponse}
   */
  @Override
  public MockResponse handleDelete(String path) {
    return withLock(lock.writeLock(), () -> handle(path, this::processDelete));
  }

  private void processDelete(String path, AttributeSet pathAttributes, AttributeSet oldAttributes) {
    String jsonStringOfResource = map.get(oldAttributes);
    final GenericKubernetesResource resource = Serialization.unmarshal(jsonStringOfResource, GenericKubernetesResource.class);
    if (resource.getFinalizers().isEmpty()) {
      // No finalizers left, actually remove the resource.
      processEvent(path, pathAttributes, oldAttributes, null, null);
      return;
    }
    if (!resource.isMarkedForDeletion()) {
      // Mark the resource as deleted, but don't remove it yet (wait for finalizer-removal).
      resource.getMetadata().setDeletionTimestamp(
          ZonedDateTime.now(ZoneOffset.UTC).truncatedTo(ChronoUnit.SECONDS).format(DateTimeFormatter.ISO_OFFSET_DATE_TIME));
      resource.getMetadata().setResourceVersion(String.valueOf(requestResourceVersion()));
      String updatedResource = Serialization.asJson(resource);
      processEvent(path, pathAttributes, oldAttributes, resource, updatedResource);
      return;
    }
    // else: if the resource is already marked for deletion and still has finalizers, do nothing.
  }

  @Override
  public long requestResourceVersion() {
    return resourceVersion.incrementAndGet();
  }

  @Override
  public AttributeSet getKey(String path) {
    return kubernetesAttributesExtractor.fromPath(path);
  }

  @Override
  public Map.Entry findResource(AttributeSet attributes) {
    // A lock (either read or write) has already been acquired by the point this method is invoked
    return map.entrySet().stream()
        .filter(entry -> entry.getKey().matches(attributes))
        .findFirst().orElse(null);
  }

  @Override
  public boolean isStatusSubresourceEnabledForResource(String path) {
    return crdProcessor.isStatusSubresourceEnabledForResource(kubernetesAttributesExtractor.fromKubernetesPath(path));
  }

  @Override
  public void processEvent(String path, AttributeSet pathAttributes, AttributeSet oldAttributes,
      GenericKubernetesResource resource, String newState) {
    // A write lock has already been acquired by the point this method is invoked
    String existing = map.remove(oldAttributes);
    AttributeSet newAttributes = null;
    if (newState != null) {
      if (resource != null) {
        newAttributes = kubernetesAttributesExtractor.extract(resource);
      } else {
        newAttributes = kubernetesAttributesExtractor.fromResource(newState);
      }
      // corner case - we need to get the plural from the path
      if (!newAttributes.containsKey(KubernetesAttributesExtractor.PLURAL)) {
        newAttributes = AttributeSet.merge(pathAttributes, newAttributes);
      }
      map.put(newAttributes, newState);
    }
    if (!Objects.equals(existing, newState)) {
      AttributeSet finalAttributeSet = newAttributes;
      watchEventListeners.forEach(listener -> {
        boolean matchesOld = oldAttributes != null && listener.attributeMatches(oldAttributes);
        boolean matchesNew = finalAttributeSet != null && listener.attributeMatches(finalAttributeSet);
        if (matchesOld && matchesNew) {
          listener.sendWebSocketResponse(newState, Action.MODIFIED);
        } else if (matchesOld) {
          listener.sendWebSocketResponse(existing, Action.DELETED);
        } else if (matchesNew) {
          listener.sendWebSocketResponse(newState, Action.ADDED);
        }
      });

      crdProcessor.process(path, Utils.getNonNullOrElse(newState, existing), newState == null);
    }
  }

  /**
   * Watch the resource list on `path` endpoint
   *
   * @param path String
   * @return The {@link MockResponse}
   */
  public MockResponse handleWatch(String path) {
    MockResponse mockResponse = new MockResponse();
    String resourceName = fetchResourceNameFromWatchRequestPath(path);
    AttributeSet query = attributeExtractor.fromPath(path);
    if (resourceName != null) {
      query = query.add(new Attribute("name", resourceName));
    }
    WatchEventsListener watchEventListener = new WatchEventsListener(context, query, watchEventListeners, LOGGER,
        watch -> withLock(lock.readLock(), () -> map.entrySet().stream()
            .filter(entry -> watch.attributeMatches(entry.getKey()))
            .forEach(entry -> watch.sendWebSocketResponse(entry.getValue(), Action.ADDED))));
    watchEventListeners.add(watchEventListener);
    return mockResponse.withWebSocketUpgrade(watchEventListener);
  }

  private boolean detectWatchMode(String path) {
    String queryString = null;
    try {
      queryString = new URI(path).getQuery();
    } catch (URISyntaxException e) {
      LOGGER.debug("incorrect URI string: [{}]", path);
      return false;
    }
    if (queryString != null && !queryString.isEmpty()) {
      return queryString.contains("watch=true");
    }
    return false;
  }

  private String fetchResourceNameFromWatchRequestPath(String path) {
    String queryString = null;
    try {
      queryString = new URI(path).getQuery();
    } catch (URISyntaxException e) {
      LOGGER.debug("Incorrect URI string: [{}]", path);
      return null;
    }

    if (queryString == null || queryString.isEmpty()) {
      return null;
    }

    String name = "";
    for (String q : queryString.split("&")) {
      if (q.contains("fieldSelector") && q.contains("metadata.name")) {
        String[] s = q.split("=");
        name = s[s.length - 1];
      }
    }
    return name.isEmpty() ? null : name;
  }

  @Override
  public void reset() {
    withLock(lock.writeLock(), map::clear);
    // what about the initial crds? That should likely be deprecated
    this.crdProcessor.reset();
  }

  @Override
  public void expectCustomResource(CustomResourceDefinitionContext rdc) {
    this.crdProcessor.addCrdContext(rdc);
  }

  /**
   * Run the given task after acquiring the lock.
   *
   * @param lock a lock to be acquired
   * @param task runnable task to execute with the acquired lock
   */
  private void withLock(Lock lock, Runnable task) {
    withLock(lock, () -> {
      task.run();
      return null;
    });
  }

  /**
   * Retrieve the value provided by the supplier after acquiring the lock.
   *
   * @param  the type supplied by source
   * @param lock a lock to be acquired
   * @param source supplier giving a return value
   * @return the value given by the source supplier
   */
  private  T withLock(Lock lock, Supplier source) {
    lock.lock();
    try {
      locked(lock);
      return source.get();
    } finally {
      lock.unlock();
    }
  }

  protected void locked(Lock lock) {
    // For testing
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy