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

io.fabric8.kubernetes.client.server.mock.crud.PatchHandler 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.crud;

import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.node.ObjectNode;
import io.fabric8.kubernetes.api.model.GenericKubernetesResource;
import io.fabric8.kubernetes.client.dsl.base.PatchType;
import io.fabric8.kubernetes.client.utils.Serialization;
import io.fabric8.kubernetes.client.utils.Utils;
import io.fabric8.mockwebserver.crud.AttributeSet;
import io.fabric8.mockwebserver.http.MediaType;
import io.fabric8.mockwebserver.http.MockResponse;
import io.fabric8.zjsonpatch.JsonPatch;

import java.net.HttpURLConnection;
import java.util.Iterator;
import java.util.Map;
import java.util.Objects;

import static io.fabric8.kubernetes.client.server.mock.crud.KubernetesCrudDispatcherHandler.isStatusPath;
import static io.fabric8.kubernetes.client.server.mock.crud.KubernetesCrudDispatcherHandler.setStatus;
import static java.net.HttpURLConnection.HTTP_ACCEPTED;
import static java.net.HttpURLConnection.HTTP_UNSUPPORTED_TYPE;

public class PatchHandler implements KubernetesCrudDispatcherHandler {

  private static final String PATH = "path";

  private final KubernetesCrudPersistence persistence;

  public PatchHandler(KubernetesCrudPersistence persistence) {
    this.persistence = persistence;
  }

  @Override
  public MockResponse handle(String path, String contentType, String requestBody) throws KubernetesCrudDispatcherException {
    final AttributeSet query = persistence.getKey(path);

    final Map.Entry currentResourceEntry = persistence.findResource(query);
    if (currentResourceEntry == null) {
      return new MockResponse().setResponseCode(HttpURLConnection.HTTP_NOT_FOUND);
    }

    final JsonNode currentResource = persistence.asNode(currentResourceEntry);
    final JsonNode patch = persistence.asNode(requestBody);
    // Read the patch and create a complete resource (either from the body or by applying the PATCH operations)
    final JsonNode fullPatch;
    if (getMergeType(contentType) == PatchType.JSON) {
      fullPatch = JsonPatch.apply(patch, initPaths(currentResource.deepCopy(), patch));
    } else {
      fullPatch = persistence.merge(currentResource, requestBody);
    }
    validatePath(query, fullPatch);
    validateResourceVersion(currentResource, fullPatch);

    final JsonNode updatedResource;
    if (isStatusPath(path)) {
      updatedResource = currentResource.deepCopy();
      setStatus(updatedResource, fullPatch.get(STATUS));
    } else {
      updatedResource = fullPatch;
      // preserve original status (PATCH requests to the custom resource ignore changes to the status stanza)
      if (persistence.isStatusSubresourceEnabledForResource(path)) {
        setStatus(updatedResource, currentResource.path(STATUS));
      }
    }

    persistence.preserveMetadata(currentResource, updatedResource);
    if (!isStatusPath(path)) {
      persistence.touchGeneration(currentResource, updatedResource);
    }
    persistence.touchResourceVersion(currentResource, updatedResource);
    String updatedAsString = Serialization.asJson(updatedResource);
    GenericKubernetesResource deserializedResource = validateRequestBody(updatedAsString);
    if (deserializedResource.isMarkedForDeletion() && deserializedResource.getFinalizers().isEmpty()) {
      // Delete the resource.
      updatedAsString = null;
      deserializedResource = null;
    }

    persistence.processEvent(path, query, currentResourceEntry.getKey(), deserializedResource, updatedAsString);
    return new MockResponse().setResponseCode(HTTP_ACCEPTED).setBody(Utils.getNonNullOrElse(updatedAsString, ""));
  }

  private PatchType getMergeType(String contentType) throws KubernetesCrudDispatcherException {
    final PatchType mergeType;
    if (contentType == null) {
      mergeType = PatchType.JSON;
    } else {
      final String subtype = Objects.requireNonNull(MediaType.parse(contentType)).subtype();
      if (subtype.equals(MediaType.get(PatchType.JSON.getContentType()).subtype())) {
        mergeType = PatchType.JSON;
      } else if (subtype.equals(MediaType.get(PatchType.JSON_MERGE.getContentType()).subtype())) {
        mergeType = PatchType.JSON_MERGE;
      } else {
        throw new KubernetesCrudDispatcherException("Unsupported Media Type", HTTP_UNSUPPORTED_TYPE);
      }
    }
    return mergeType;
  }

  // Ensure resource contains all paths affected by the provided patch
  private JsonNode initPaths(JsonNode resource, JsonNode patch) {
    for (Iterator it = patch.elements(); it.hasNext();) {
      final String fullPath = it.next().get(PATH).asText();
      final String[] paths = fullPath.replaceAll("^/", "").split("/");
      JsonNode node = resource;
      for (int p = 0; p < paths.length - 1; p++) {
        final String path = paths[p];
        if (node.get(path) == null && node.isObject()) {
          ((ObjectNode) node).set(path, ((ObjectNode) node).objectNode());
        }
        node = node.path(path);
      }
    }
    return resource;
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy