Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Port changes from 4.10 #21

Merged
merged 3 commits into from
Dec 26, 2021
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ This is a port of [java-diff-utils](https://github.com/java-diff-utils/java-diff
with multiplatform support. All credit for the implementation goes to original authors.

## Features
All features from version 4.9 of the original library are present, except for:
All features from version 4.10 of the original library are present, except for:
* Unified diff, which heavily uses file read/write and therefore needs a more complicated rewrite for kotlin-multiplatform
* diff-utils-jgit, which uses JVM-only jgit library

Expand Down
20 changes: 10 additions & 10 deletions src/commonMain/kotlin/io/github/petertrr/diffutils/patch/Chunk.kt
Original file line number Diff line number Diff line change
Expand Up @@ -46,17 +46,17 @@ public data class Chunk<T>(
* @param target the sequence to verify against.
* @throws PatchFailedException
*/
@Throws(PatchFailedException::class)
public fun verify(target: List<T>) {
if (position > target.size || last() > target.size) {
throw PatchFailedException("Incorrect Chunk: the position of chunk > target size")
}
for (i in 0 until size()) {
if (target[position + i] != lines[i]) {
throw PatchFailedException(
"Incorrect Chunk: the chunk content doesn't match the target"
)
public fun verify(target: List<T>): VerifyChunk {
return if (position > target.size || last() > target.size) {
VerifyChunk.POSITION_OUT_OF_TARGET
} else if (
(0 until size()).any { i ->
target[position + i] != lines[i]
}
) {
VerifyChunk.CONTENT_DOES_NOT_MATCH_TARGET
} else {
VerifyChunk.OK
}
}

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,52 @@
/*
* Copyright 2021 Peter Trifanov.
* Copyright 2021 java-diff-utils.
*
* 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.
*
* This file has been modified by Peter Trifanov when porting from Java to Kotlin.
*/
package io.github.petertrr.diffutils.patch

public fun interface ConflictOutput<T> {
@Throws(PatchFailedException::class)
public fun processConflict(verifyChunk: VerifyChunk, delta: Delta<T>, result: MutableList<T>)
}

public class ExceptionProducingConflictOutput<T> : ConflictOutput<T> {
override fun processConflict(verifyChunk: VerifyChunk, delta: Delta<T>, result: MutableList<T>) {
throw PatchFailedException(
"could not apply patch due to $verifyChunk"
)
}
}

public class ConflictProducingConflictOutput : ConflictOutput<String> {
override fun processConflict(verifyChunk: VerifyChunk, delta: Delta<String>, result: MutableList<String>) {
if (result.size > delta.source.position) {
val orgData = mutableListOf<String>()
(0 until delta.source.size()).forEach { _ ->
orgData.add(
result.removeAt(delta.source.position)
)
}
orgData.add(0, "<<<<<< HEAD")
orgData.add("======")
orgData.addAll(delta.source.lines)
orgData.add(">>>>>>> PATCH")
result.addAll(delta.source.position, orgData)
} else {
TODO("Not supported yet.")
}
}
}
26 changes: 16 additions & 10 deletions src/commonMain/kotlin/io/github/petertrr/diffutils/patch/Delta.kt
Original file line number Diff line number Diff line change
Expand Up @@ -62,12 +62,21 @@ public sealed class Delta<T>(public val type: DeltaType) {
* @throws PatchFailedException
*/
@Throws(PatchFailedException::class)
protected open fun verifyChunk(target: List<T>) {
source.verify(target)
protected open fun verifyChunkToFitTarget(target: List<T>): VerifyChunk {
return source.verify(target)
}

@Throws(PatchFailedException::class)
public abstract fun applyTo(target: MutableList<T>)
public open fun verifyAndApplyTo(target: MutableList<T>): VerifyChunk {
val verify: VerifyChunk = verifyChunkToFitTarget(target)
if (verify == VerifyChunk.OK) {
applyTo(target)
}
return verify
}

@Throws(PatchFailedException::class)
protected abstract fun applyTo(target: MutableList<T>)

public abstract fun restore(target: MutableList<T>)

Expand All @@ -77,8 +86,7 @@ public sealed class Delta<T>(public val type: DeltaType) {
public abstract fun withChunks(original: Chunk<T>, revised: Chunk<T>): Delta<T>
}
public data class ChangeDelta<T>(override val source: Chunk<T>, override val target: Chunk<T>) : Delta<T>(DeltaType.CHANGE) {
override fun applyTo(target: MutableList<T>) {
verifyChunk(target)
protected override fun applyTo(target: MutableList<T>) {
val position: Int = source.position
val size: Int = source.size()
for (i in 0 until size) {
Expand All @@ -104,8 +112,7 @@ public data class ChangeDelta<T>(override val source: Chunk<T>, override val tar
}

public data class DeleteDelta<T>(override val source: Chunk<T>, override val target: Chunk<T>) : Delta<T>(DeltaType.DELETE) {
override fun applyTo(target: MutableList<T>) {
verifyChunk(target)
protected override fun applyTo(target: MutableList<T>) {
val position = source.position
for (i in 0 until source.size()) {
target.removeAt(position)
Expand All @@ -124,8 +131,7 @@ public data class DeleteDelta<T>(override val source: Chunk<T>, override val tar
}

public data class InsertDelta<T>(override val source: Chunk<T>, override val target: Chunk<T>) : Delta<T>(DeltaType.INSERT) {
override fun applyTo(target: MutableList<T>) {
verifyChunk(target)
protected override fun applyTo(target: MutableList<T>) {
val position = this.source.position
this.target.lines.forEachIndexed { i, line ->
target.add(position + i, line)
Expand All @@ -143,7 +149,7 @@ public data class InsertDelta<T>(override val source: Chunk<T>, override val tar
}

public data class EqualDelta<T>(override val source: Chunk<T>, override val target: Chunk<T>) : Delta<T>(DeltaType.EQUAL) {
override fun applyTo(target: MutableList<T>): Unit = verifyChunk(target)
protected override fun applyTo(target: MutableList<T>): Unit = Unit

override fun restore(target: MutableList<T>): Unit = Unit

Expand Down
13 changes: 11 additions & 2 deletions src/commonMain/kotlin/io/github/petertrr/diffutils/patch/Patch.kt
Original file line number Diff line number Diff line change
Expand Up @@ -24,8 +24,11 @@ import io.github.petertrr.diffutils.algorithm.Change
* Describes the patch holding all deltas between the original and revised texts.
*
* @param T The type of the compared elements in the 'lines'.
* @property conflictOutput Alter normal conflict output behaviour to e.g. inclide some conflict statements in the result, like git does it.
*/
public class Patch<T> {
public class Patch<T>(
private var conflictOutput: ConflictOutput<T> = ExceptionProducingConflictOutput(),
) {
public var deltas: MutableList<Delta<T>> = arrayListOf()
get() {
field.sortBy { it.source.position }
Expand All @@ -43,7 +46,9 @@ public class Patch<T> {
val it = deltas.listIterator(deltas.size)
while (it.hasPrevious()) {
val delta = it.previous()
delta.applyTo(result)
delta.verifyAndApplyTo(result).takeIf { it != VerifyChunk.OK }?.let {
conflictOutput.processConflict(it, delta, result)
}
}
return result
}
Expand Down Expand Up @@ -75,6 +80,10 @@ public class Patch<T> {
return "Patch{deltas=$deltas}"
}

public fun withConflictOutput(conflictOutput: ConflictOutput<T>) {
this.conflictOutput = conflictOutput
}

public companion object {
public fun <T> generate(original: List<T>, revised: List<T>, changes: List<Change>): Patch<T> {
return generate(original, revised, changes, false)
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
/*
* Copyright 2021 Peter Trifanov.
* Copyright 2021 java-diff-utils.
*
* 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.
*
* This file has been modified by Peter Trifanov when porting from Java to Kotlin.
*/
package io.github.petertrr.diffutils.patch

public enum class VerifyChunk {
OK,
POSITION_OUT_OF_TARGET,
CONTENT_DOES_NOT_MATCH_TARGET,
;
}
Original file line number Diff line number Diff line change
Expand Up @@ -60,4 +60,34 @@ class PatchTest {
fail(e.message)
}
}

@Test
fun testPatch_Change_withExceptionProcessor() {
val changeTest_from = listOf("aaa", "bbb", "ccc", "ddd")
val changeTest_to = listOf("aaa", "bxb", "cxc", "ddd")
val patch: Patch<String> = diff(changeTest_from, changeTest_to)
patch.withConflictOutput(ConflictProducingConflictOutput())
try {
val data: List<String> = patch(
changeTest_from.toMutableList().apply { this[2] = "CDC" },
patch
)
assertEquals(9, data.size)
assertEquals(
listOf(
"aaa",
"<<<<<< HEAD",
"bbb",
"CDC",
"======",
"bbb",
"ccc",
">>>>>>> PATCH",
"ddd"
), data
)
} catch (e: PatchFailedException) {
fail(e.message)
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -683,4 +683,35 @@ Bengal tiger panther but singapura but bombay munchkin for cougar. And more.""".
)
assertEquals(DiffRow(DiffRow.Tag.DELETE, "~B~", ""), rows[1])
}

@Test
fun testIssue119WrongContextLength() {
val original: String =
"""
const world: string = 'world',
p: number | undefined = 42;

console.log(`Hello, ${'$'}world}!`);
""".trimIndent()
val revised: String =
"""
const world: string = 'world';
const p: number | undefined = 42;

console.log(`Hello, ${'$'}world}!`);
""".trimIndent()
val generator = DiffRowGenerator(
showInlineDiffs = true,
mergeOriginalRevised = true,
inlineDiffByWord = true,
oldTag = { _, _ -> "~" },
newTag = { _, _ -> "**" },
)
val rows: List<DiffRow> = generator.generateDiffRows(
original.split("\n"),
revised.split("\n")
)
rows.filter { it.tag != DiffRow.Tag.EQUAL }
.forEach { println(it) }
}
}