Skip to content

Commit

Permalink
Merge pull request #11 from goatfryed/feat/#3/assert-serializable
Browse files Browse the repository at this point in the history
feat(#3): add support for java's Serializable
  • Loading branch information
goatfryed authored Oct 26, 2024
2 parents 461955e + c6ea140 commit 0e52336
Show file tree
Hide file tree
Showing 10 changed files with 208 additions and 49 deletions.
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
package io.github.goatfryed.assert_baseline;

import io.github.goatfryed.assert_baseline.json.JsonBaselineAssertion;
import io.github.goatfryed.assert_baseline.serializable.SerializableBaselineAssertion;
import io.github.goatfryed.assert_baseline.text.TextBaselineAssertion;
import io.github.goatfryed.assert_baseline.xml.XmlBaselineAssertion;

Expand All @@ -10,11 +11,15 @@ public static JsonBaselineAssertion assertThatJson(String string) {
return new JsonBaselineAssertion(new SerializableSubject(string));
}

public static TextBaselineAssertion assertThatText(String string) {
return new TextBaselineAssertion(new SerializableSubject(string));
}

public static XmlBaselineAssertion assertThatXml(String string) {
return new XmlBaselineAssertion(new SerializableSubject(string));
}

public static TextBaselineAssertion assertThatText(String string) {
return new TextBaselineAssertion(new SerializableSubject(string));
public static <ACTUAL> SerializableBaselineAssertion<ACTUAL> assertThatSerializable(ACTUAL actual) {
return new SerializableBaselineAssertion<>(actual);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -8,14 +8,14 @@

import java.util.function.Function;

abstract public class AbstractBaselineAssertion<SELF extends AbstractBaselineAssertion<SELF>>
extends AbstractAssert<SELF, String>
abstract public class AbstractBaselineAssertion<SELF extends AbstractBaselineAssertion<SELF,ACTUAL>, ACTUAL>
extends AbstractAssert<SELF, ACTUAL>
implements BaselineAssertion<SELF> {

private BaselineContextFactory contextFactory;

protected AbstractBaselineAssertion(String string, Class<?> selfType) {
super(string, selfType);
protected AbstractBaselineAssertion(ACTUAL actual, Class<?> selfType) {
super(actual, selfType);
}

public SELF using(Function<BaselineContextFactory,BaselineContextFactory> config) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,8 @@

import static net.javacrumbs.jsonunit.assertj.JsonAssertions.assertThatJson;

public class JsonBaselineAssertion extends AbstractBaselineAssertion<JsonBaselineAssertion> implements BaselineAssertionAdapter {
public class JsonBaselineAssertion extends AbstractBaselineAssertion<JsonBaselineAssertion,String> {

private final SerializableSubject subject;
private Function<Configuration, Configuration> comparatorConfigurer = Function.identity();

Expand Down Expand Up @@ -46,25 +47,27 @@ public JsonBaselineAssertion usingJsonComparator(
return myself;
}

@Override
public void writeActual(BaselineContext.ActualOutput actualOutput, BaselineContext context) {
subject.writeTo(actualOutput::outputStream);
}

@Override
public void assertEquals(BaselineContext.BaselineInput baselineInput, BaselineContext context) {
getJsonAssert()
.describedAs(context.asDescription())
.isEqualTo(baselineInput.readContentAsString());
}

private JsonAssert.ConfigurableJsonAssert getJsonAssert() {
return assertThatJson(subject.serialized())
.withConfiguration(comparatorConfigurer);
}

@Override
protected @NotNull BaselineAssertionAdapter getAssertionAdapter() {
return this;
return new Adapter();
}

private class Adapter implements BaselineAssertionAdapter {
@Override
public void writeActual(BaselineContext.ActualOutput actualOutput, BaselineContext context) {
subject.writeTo(actualOutput::outputStream);
}

@Override
public void assertEquals(BaselineContext.BaselineInput baselineInput, BaselineContext context) {
getJsonAssert()
.describedAs(context.asDescription())
.isEqualTo(baselineInput.readContentAsString());
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,76 @@
package io.github.goatfryed.assert_baseline.serializable;

import io.github.goatfryed.assert_baseline.core.AbstractBaselineAssertion;
import io.github.goatfryed.assert_baseline.core.BaselineAssertionAdapter;
import io.github.goatfryed.assert_baseline.core.BaselineContext;
import org.assertj.core.api.Assert;
import org.assertj.core.api.ObjectAssert;
import org.jetbrains.annotations.NotNull;

import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.util.function.Consumer;
import java.util.function.Function;

import static org.assertj.core.api.Assertions.assertThat;

public class SerializableBaselineAssertion<ACTUAL> extends AbstractBaselineAssertion<SerializableBaselineAssertion<ACTUAL>,ACTUAL> {

private Function<ObjectAssert<ACTUAL>, Assert<?,?>> assertConfig = assertion -> assertion;

public SerializableBaselineAssertion(ACTUAL actual) {
super(actual, SerializableBaselineAssertion.class);
}

/**
* This is the same as just calling<br>
* <code>
* serializableAssertion.satisfies(it -> assertThat(it)...)
* </code><br>
* It's only implemented as a convenience method in line with our general pattern
*/
public SerializableBaselineAssertion<ACTUAL> serializableSatisfies(Consumer<ObjectAssert<ACTUAL>> assertion) {
assertion.accept(assertThat(actual));
return myself;
}

/**
* This yields control of the {@link Assert} to the user.<br>
* Note that whatever assertion is returned gets called with <code>isEqualTo(baselineObject)</code>.
*/
public SerializableBaselineAssertion<ACTUAL> usingSerializableComparator(Function<ObjectAssert<ACTUAL>, Assert<?,?>> assertConfig) {
this.assertConfig = assertConfig;
return myself;
}

@Override
protected @NotNull BaselineAssertionAdapter getAssertionAdapter() {
return new Adapter();
}

class Adapter implements BaselineAssertionAdapter {
@Override
public void writeActual(BaselineContext.ActualOutput output, BaselineContext context) {
try (var oos = new ObjectOutputStream(output.outputStream())) {
oos.writeObject(actual);
} catch (IOException e) {
throw new AssertionError("failed to serialize actual", e);
}
}

@Override
public void assertEquals(BaselineContext.BaselineInput baseline, BaselineContext context) {
try (var ois = new ObjectInputStream(baseline.getInputStream())) {
var baselineObject = ois.readObject();
assertConfig.apply(assertThat(actual))
.isEqualTo(baselineObject);
} catch (IOException e) {
throw new AssertionError("failed to read baseline", e);
} catch (ClassNotFoundException e) {
throw new AssertionError("baseline class does not exist. This indicates that you changed class definition and need to recreate the baseline.", e);
}
}
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@

import static org.assertj.core.api.Assertions.assertThat;

public class TextBaselineAssertion extends AbstractBaselineAssertion<TextBaselineAssertion> implements BaselineAssertionAdapter {
public class TextBaselineAssertion extends AbstractBaselineAssertion<TextBaselineAssertion,String> {

private final SerializableSubject subject;

Expand All @@ -27,22 +27,25 @@ public TextBaselineAssertion textSatisfies(Consumer<AbstractStringAssert<?>> ass

@Override
protected @NotNull BaselineAssertionAdapter getAssertionAdapter() {
return this;
return new Adapter();
}

@Override
public void writeActual(BaselineContext.ActualOutput actualOutput, BaselineContext context) {
subject.writeTo(actualOutput::outputStream);
private AbstractStringAssert<?> getStringAssert() {
return assertThat(subject.serialized());
}

@Override
public void assertEquals(BaselineContext.BaselineInput baselineInput, BaselineContext context) {
getStringAssert()
.describedAs(context.asDescription())
.isEqualTo(baselineInput.readContentAsString());
}
private class Adapter implements BaselineAssertionAdapter {

private AbstractStringAssert<?> getStringAssert() {
return assertThat(subject.serialized());
@Override
public void writeActual(BaselineContext.ActualOutput actualOutput, BaselineContext context) {
subject.writeTo(actualOutput::outputStream);
}

@Override
public void assertEquals(BaselineContext.BaselineInput baselineInput, BaselineContext context) {
getStringAssert()
.describedAs(context.asDescription())
.isEqualTo(baselineInput.readContentAsString());
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,8 @@

import static org.xmlunit.assertj.XmlAssert.assertThat;

public class XmlBaselineAssertion extends AbstractBaselineAssertion<XmlBaselineAssertion> implements BaselineAssertionAdapter {
public class XmlBaselineAssertion extends AbstractBaselineAssertion<XmlBaselineAssertion,String> {

private final SerializableSubject subject;
private Function<CompareAssert, CompareAssert> comparatorConfigurer = Function.identity();

Expand Down Expand Up @@ -48,24 +49,27 @@ public XmlBaselineAssertion usingXmlComparator(

@Override
protected @NotNull BaselineAssertionAdapter getAssertionAdapter() {
return this;
return new Adapter();
}

@Override
public void writeActual(BaselineContext.ActualOutput actualOutput, BaselineContext context) {
subject.writeTo(actualOutput::outputStream);
private XmlAssert getXmlAssert() {
return assertThat(subject.serialized());
}

@Override
public void assertEquals(BaselineContext.BaselineInput baselineInput, BaselineContext context) {
comparatorConfigurer.apply(
getXmlAssert().and(baselineInput.readContentAsString())
)
.describedAs(context.asDescription())
.areSimilar();
}
private class Adapter implements BaselineAssertionAdapter {

private XmlAssert getXmlAssert() {
return assertThat(subject.serialized());
@Override
public void writeActual(BaselineContext.ActualOutput actualOutput, BaselineContext context) {
subject.writeTo(actualOutput::outputStream);
}

@Override
public void assertEquals(BaselineContext.BaselineInput baselineInput, BaselineContext context) {
comparatorConfigurer.apply(
getXmlAssert().and(baselineInput.readContentAsString())
)
.describedAs(context.asDescription())
.areSimilar();
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ public void itValidatesThatActualAndBaselineAreDifferent() {
).hasMessageContaining("Both should have different names");
}

private static class DummyBaselineAssertion extends AbstractBaselineAssertion<DummyBaselineAssertion> {
private static class DummyBaselineAssertion extends AbstractBaselineAssertion<DummyBaselineAssertion,String> {

protected DummyBaselineAssertion() {
super("ignored", DummyBaselineAssertion.class);
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
package io.github.goatfryed.assert_baseline.serializable;

import java.io.Serializable;

public record Foo(String foo, Bar fooBar) implements Serializable {

public record Bar(String prop1, String prop2) implements Serializable { }
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,60 @@
package io.github.goatfryed.assert_baseline.serializable;

import org.junit.jupiter.api.Test;

import java.io.File;
import java.nio.file.Path;

import static io.github.goatfryed.assert_baseline.BaselineAssertions.assertThatSerializable;
import static org.assertj.core.api.AssertionsForClassTypes.assertThatCode;

class SerializableBaselineAssertionTest {

public final static String BASE_PATH = "serializable/object.%s.json";
public final static String ALTERED_PATH = BASE_PATH.formatted("altered");
public final static String ACTUAL_PATH = BASE_PATH.formatted("actual");
public final static String BASELINE_PATH = BASE_PATH.formatted("baseline");
public final static Path RESOURCE_ROOT = Path.of("src/test/resources");
public final static File ACTUAL_FILE = RESOURCE_ROOT.resolve(ACTUAL_PATH).toFile();
public final static File ALTERED_FILE = RESOURCE_ROOT.resolve(ALTERED_PATH).toFile();
public final static File BASELINE_FILE = RESOURCE_ROOT.resolve(BASELINE_PATH).toFile();

@Test
void itPassesSameContent() {
var foo = new Foo("chucky", new Foo.Bar("chuck","norris"));

assertThatSerializable(foo)
.isEqualToBaseline(BASELINE_PATH);
}

@Test
void itFailsContentViolations() {
var foo = new Foo("chucky", new Foo.Bar("chuck","the murder puppet"));

assertThatCode(() ->
assertThatSerializable(foo)
.isEqualToBaseline(BASELINE_PATH)
).hasMessageContaining("expected: Foo[foo=chucky, fooBar=Bar[prop1=chuck, prop2=norris]]");
}

@Test
void itPassesAllowedDifferences() {
var foo = new Foo("chucky", new Foo.Bar("chuck","the murder puppet"));

assertThatSerializable(foo)
.usingSerializableComparator(assertion -> assertion
.usingRecursiveComparison()
.ignoringFields("foo", "fooBar.prop2")
).isEqualToBaseline(BASELINE_PATH);
}

@Test
void itInteropsWithAssertJ() {
var foo = new Foo("chucky", new Foo.Bar("chuck","norris"));

assertThatSerializable(foo)
.serializableSatisfies(assertion -> assertion
.hasFieldOrProperty("fooBar")
);
}
}
Binary file added src/test/resources/serializable/object.baseline.json
Binary file not shown.

0 comments on commit 0e52336

Please sign in to comment.