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

com.linecorp.centraldogma.common.Change Maven / Gradle / Ivy

Go to download

Highly-available version-controlled service configuration repository based on Git, ZooKeeper and HTTP/2 (centraldogma-common)

There is a newer version: 0.72.0
Show newest version
/*
 * Copyright 2017 LINE Corporation
 *
 * LINE Corporation licenses this file to you 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:
 *
 *   https://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.linecorp.centraldogma.common;

import static com.linecorp.centraldogma.internal.Util.validateDirPath;
import static com.linecorp.centraldogma.internal.Util.validateFilePath;
import static java.util.Objects.requireNonNull;

import java.io.File;
import java.io.IOError;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.Collections;
import java.util.List;
import java.util.stream.Collectors;
import java.util.stream.Stream;

import javax.annotation.Nullable;

import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.annotation.JsonDeserialize;

import com.linecorp.centraldogma.internal.Jackson;
import com.linecorp.centraldogma.internal.Util;
import com.linecorp.centraldogma.internal.jsonpatch.JsonPatch;
import com.linecorp.centraldogma.internal.jsonpatch.ReplaceMode;

import difflib.DiffUtils;
import difflib.Patch;

/**
 * A modification of an individual {@link Entry}.
 */
@JsonDeserialize(as = DefaultChange.class)
public interface Change {

    /**
     * Returns a newly-created {@link Change} whose type is {@link ChangeType#UPSERT_TEXT}.
     *
     * 

Note that you should use {@link #ofJsonUpsert(String, String)} if the specified {@code path} ends with * {@code ".json"}. The {@link #ofJsonUpsert(String, String)} will check that the given {@code text} is a * valid JSON. * * @param path the path of the file * @param text the content of the file * @throws ChangeFormatException if the path ends with {@code ".json"} */ static Change ofTextUpsert(String path, String text) { requireNonNull(text, "text"); validateFilePath(path, "path"); if (EntryType.guessFromPath(path) == EntryType.JSON) { throw new ChangeFormatException("invalid file type: " + path + " (expected: a non-JSON file). Use Change.ofJsonUpsert() instead"); } return new DefaultChange<>(path, ChangeType.UPSERT_TEXT, text); } /** * Returns a newly-created {@link Change} whose type is {@link ChangeType#UPSERT_JSON}. * * @param path the path of the file * @param jsonText the content of the file * * @throws ChangeFormatException if the specified {@code jsonText} is not a valid JSON */ static Change ofJsonUpsert(String path, String jsonText) { requireNonNull(jsonText, "jsonText"); final JsonNode jsonNode; try { jsonNode = Jackson.readTree(jsonText); } catch (IOException e) { throw new ChangeFormatException("failed to read a value as a JSON tree", e); } return new DefaultChange<>(path, ChangeType.UPSERT_JSON, jsonNode); } /** * Returns a newly-created {@link Change} whose type is {@link ChangeType#UPSERT_JSON}. * * @param path the path of the file * @param jsonNode the content of the file */ static Change ofJsonUpsert(String path, JsonNode jsonNode) { requireNonNull(jsonNode, "jsonNode"); return new DefaultChange<>(path, ChangeType.UPSERT_JSON, jsonNode); } /** * Returns a newly-created {@link Change} whose type is {@link ChangeType#REMOVE}. * * @param path the path of the file to remove */ static Change ofRemoval(String path) { return new DefaultChange<>(path, ChangeType.REMOVE, null); } /** * Returns a newly-created {@link Change} whose type is {@link ChangeType#RENAME}. * * @param oldPath the old path of the file * @param newPath the new path of the file */ static Change ofRename(String oldPath, String newPath) { validateFilePath(oldPath, "oldPath"); validateFilePath(newPath, "newPath"); return new DefaultChange<>(oldPath, ChangeType.RENAME, newPath); } /** * Returns a newly-created {@link Change} whose type is {@link ChangeType#APPLY_TEXT_PATCH}. * *

Note that you should use {@link #ofJsonPatch(String, String, String)} if the specified {@code path} * ends with {@code ".json"}. The {@link #ofJsonUpsert(String, String)} will check that * the given {@code oldText} and {@code newText} are valid JSONs. * * @param path the path of the file * @param oldText the old content of the file * @param newText the new content of the file * @throws ChangeFormatException if the path ends with {@code ".json"} */ static Change ofTextPatch(String path, @Nullable String oldText, String newText) { validateFilePath(path, "path"); requireNonNull(newText, "newText"); if (EntryType.guessFromPath(path) == EntryType.JSON) { throw new ChangeFormatException("invalid file type: " + path + " (expected: a non-JSON file). Use Change.ofJsonPatch() instead"); } final List oldLineList = oldText == null ? Collections.emptyList() : Util.stringToLines(oldText); final List newLineList = Util.stringToLines(newText); final Patch patch = DiffUtils.diff(oldLineList, newLineList); final List unifiedDiff = DiffUtils.generateUnifiedDiff(path, path, oldLineList, patch, 3); return new DefaultChange<>(path, ChangeType.APPLY_TEXT_PATCH, String.join("\n", unifiedDiff)); } /** * Returns a newly-created {@link Change} whose type is {@link ChangeType#APPLY_TEXT_PATCH}. * *

Note that you should use {@link #ofJsonPatch(String, String)} if the specified {@code path} * ends with {@code ".json"}. The {@link #ofJsonUpsert(String, String)} will check that * the given {@code textPatch} is a valid JSON. * * @param path the path of the file * @param textPatch the patch in * unified format * @throws ChangeFormatException if the path ends with {@code ".json"} */ static Change ofTextPatch(String path, String textPatch) { validateFilePath(path, "path"); requireNonNull(textPatch, "textPatch"); if (EntryType.guessFromPath(path) == EntryType.JSON) { throw new ChangeFormatException("invalid file type: " + path + " (expected: a non-JSON file). Use Change.ofJsonPatch() instead"); } return new DefaultChange<>(path, ChangeType.APPLY_TEXT_PATCH, textPatch); } /** * Returns a newly-created {@link Change} whose type is {@link ChangeType#APPLY_JSON_PATCH}. * * @param path the path of the file * @param oldJsonText the old content of the file * @param newJsonText the new content of the file * * @throws ChangeFormatException if the specified {@code oldJsonText} or {@code newJsonText} is * not a valid JSON */ static Change ofJsonPatch(String path, @Nullable String oldJsonText, String newJsonText) { requireNonNull(newJsonText, "newJsonText"); final JsonNode oldJsonNode; final JsonNode newJsonNode; try { oldJsonNode = oldJsonText == null ? Jackson.nullNode : Jackson.readTree(oldJsonText); newJsonNode = Jackson.readTree(newJsonText); } catch (IOException e) { throw new ChangeFormatException("failed to read a value as a JSON tree", e); } return new DefaultChange<>(path, ChangeType.APPLY_JSON_PATCH, JsonPatch.generate(oldJsonNode, newJsonNode, ReplaceMode.SAFE).toJson()); } /** * Returns a newly-created {@link Change} whose type is {@link ChangeType#APPLY_JSON_PATCH}. * * @param path the path of the file * @param oldJsonNode the old content of the file * @param newJsonNode the new content of the file */ static Change ofJsonPatch(String path, @Nullable JsonNode oldJsonNode, JsonNode newJsonNode) { requireNonNull(newJsonNode, "newJsonNode"); if (oldJsonNode == null) { oldJsonNode = Jackson.nullNode; } return new DefaultChange<>(path, ChangeType.APPLY_JSON_PATCH, JsonPatch.generate(oldJsonNode, newJsonNode, ReplaceMode.SAFE).toJson()); } /** * Returns a newly-created {@link Change} whose type is {@link ChangeType#APPLY_JSON_PATCH}. * * @param path the path of the file * @param jsonPatchText the patch in JSON patch format * * @throws ChangeFormatException if the specified {@code jsonPatchText} is not a valid JSON */ static Change ofJsonPatch(String path, String jsonPatchText) { requireNonNull(jsonPatchText, "jsonPatchText"); final JsonNode jsonPatchNode; try { jsonPatchNode = Jackson.readTree(jsonPatchText); } catch (IOException e) { throw new ChangeFormatException("failed to read a value as a JSON tree", e); } return ofJsonPatch(path, jsonPatchNode); } /** * Returns a newly-created {@link Change} whose type is {@link ChangeType#APPLY_JSON_PATCH}. * * @param path the path of the file * @param jsonPatchNode the patch in JSON patch format */ static Change ofJsonPatch(String path, JsonNode jsonPatchNode) { requireNonNull(jsonPatchNode, "jsonPatchNode"); return new DefaultChange<>(path, ChangeType.APPLY_JSON_PATCH, jsonPatchNode); } /** * Creates a {@link List} of upsert {@link Change}s from all files under the specified directory * recursively. * * @param sourcePath the path to the import directory * @param targetPath the target directory path of the imported {@link Change}s * * @throws IOError if I/O error occurs */ static List> fromDirectory(Path sourcePath, String targetPath) { requireNonNull(sourcePath, "sourcePath"); validateDirPath(targetPath, "targetPath"); if (!Files.isDirectory(sourcePath)) { throw new IllegalArgumentException("sourcePath: " + sourcePath + " (must be a directory)"); } final String finalTargetPath; if (!targetPath.endsWith("/")) { finalTargetPath = targetPath + '/'; } else { finalTargetPath = targetPath; } try (Stream s = Files.find(sourcePath, Integer.MAX_VALUE, (p, a) -> a.isRegularFile())) { final int baseLength = sourcePath.toString().length() + 1; return s.map(sourceFilePath -> { final String targetFilePath = finalTargetPath + sourceFilePath.toString().substring(baseLength).replace(File.separatorChar, '/'); return fromFile(sourceFilePath, targetFilePath); }).collect(Collectors.toList()); } catch (IOException e) { throw new IOError(e); } } /** * Creates a new {@link Change} from the file at the specified location. * * @param sourcePath the path to the regular file to import * @param targetPath the target path of the imported {@link Change} */ static Change fromFile(Path sourcePath, String targetPath) { requireNonNull(sourcePath, "sourcePath"); validateFilePath(targetPath, "targetPath"); if (!Files.isRegularFile(sourcePath)) { throw new IllegalArgumentException("sourcePath: " + sourcePath + " (must be a regular file)"); } if (targetPath.endsWith("/")) { throw new IllegalArgumentException("targetPath: " + targetPath + " (must be a regular file path)"); } final EntryType entryType = EntryType.guessFromPath(targetPath); final String content; try { content = new String(Files.readAllBytes(sourcePath), StandardCharsets.UTF_8); } catch (IOException e) { throw new IOError(e); } switch (entryType) { case JSON: return ofJsonUpsert(targetPath, content); case TEXT: return ofTextUpsert(targetPath, content); default: throw new Error("unexpected entry type: " + entryType); } } /** * Returns the type of the {@link Change}. */ @JsonProperty ChangeType type(); /** * Returns the path of the {@link Change}. */ @JsonProperty String path(); /** * Returns the content of the {@link Change}, which depends on the {@link #type()}. */ @Nullable @JsonProperty T content(); /** * Returns the textual representation of {@link #content()}. */ @Nullable String contentAsText(); }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy