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

Fuzzer should change objects with its public setters #439

Merged
merged 3 commits into from
Jul 12, 2022
Merged

Conversation

Markoutte
Copy link
Collaborator

Description

Fuzzing now can create object with empty constructor and change that object with public field or field setter if exists.
Setter can have any return type but must have corresponding getter.

Fixes #289

Type of Change

  • New feature (non-breaking change which adds functionality)

How Has This Been Tested?

Automated Testing

org.utbot.framework.plugin.api.ModelProviderTest#test complex object is created with setters

Manual Scenario

Try to generate test for method equalsTo from this class:

public class MyJavaObject {

    public static int pubStaticField;
    public final int pubFinalField = 0;
    public int pubField;
    public int pubFieldWithSetter;
    private int prvField;
    private int prvFieldWithSetter;

    public int getPubFieldWithSetter() {
        return pubFieldWithSetter;
    }

    public void setPubFieldWithSetter(int pubFieldWithSetter) {
        this.pubFieldWithSetter = pubFieldWithSetter;
    }

    public int getPrvFieldWithSetter() {
        return prvFieldWithSetter;
    }

    public void setPrvFieldWithSetter(int prvFieldWithSetter) {
        this.prvFieldWithSetter = prvFieldWithSetter;
    }

    public boolean equalsTo(MyJavaObject o) {
        if (o.prvField > 0) {}
        if (this == o) return true;
        if (pubStaticField != o.pubStaticField) return false;
        if (pubFinalField != o.pubFinalField) return false;
        if (pubField != o.pubField) return false;
        if (pubFieldWithSetter != o.pubFieldWithSetter) return false;
        if (prvField != o.prvField) return false;
        if (prvFieldWithSetter != o.prvFieldWithSetter) return false;
        return true;
    }
}

Example of result test:

@Test
@DisplayName("equalsTo: o = MyJavaObject() -> return true")
public void testEqualsToReturnsTrue() {
    MyJavaObject myJavaObject = new MyJavaObject();
    MyJavaObject myJavaObject1 = new MyJavaObject();
    myJavaObject1.pubField = 0;
    myJavaObject1.setPubFieldWithSetter(0);
    myJavaObject1.setPrvFieldWithSetter(0);

    boolean actual = myJavaObject.equalsTo(myJavaObject1);

    assertTrue(actual);
}

Checklist (remove irrelevant options):

  • The change followed the style guidelines of the UTBot project
  • Self-review of the code is passed
  • The change contains enough commentaries, particularly in hard-to-understand areas
  • New documentation is provided or existed one is altered
  • No new warnings
  • All tests pass locally with my changes

@Markoutte Markoutte requested review from dtim and denis-fokin July 6, 2022 12:43
@dtim dtim requested a review from EgorkaKulikov July 7, 2022 09:34
fieldValues.asSequence().mapIndexedNotNull { index, value ->
val field = fields[index]
when {
field.setter != null -> UtExecutableCallModel(
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We prefer direct accessors in AssembleModelGenerator.chooseModificator

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

At the moment fuzzing depends on only framework-api module. Therefore, AssembleModelGenerator.chooseModificator can be accessed from fuzzing module.

FieldDescription(
field.name,
field.type.id,
field.isPublic && !field.isFinal && !field.isStatic,
Copy link
Collaborator

@EgorkaKulikov EgorkaKulikov Jul 11, 2022

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why only public, not package private, it sometimes also can be set directly

null
}
}
private val Field.isPublic
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why do you prefer to do all this logic on a Field, not on a FieldId? Related to top level methods too.

it has Modifier.PUBLIC &&
it.name == setterName &&
it.parameterCount == 1 &&
it.parameterTypes[0] == field.type
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Possibly we should enlarge StatementsStorage.isSetterOrDirectAccessor method with this type check.

private val Field.isStatic
get() = has(Modifier.STATIC)

private infix fun Field.has(modifier: Int) = (modifiers and modifier) != 0
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I do not understand what happens in this line and in the next one...

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The function has takes field.modifiers as integer with bitmask for a concreate modifier flag (STATIC in this sample) and then compares it with 0. If it is 0 then the flag wasn't set in field.modifiers. In Java it looks like:

field.getModifiers() & Modifier.STATIC != 0

That checks that this field is static.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I replaced it with method isAccessible(modifiers: Int, packageName: String?)

Copy link
Collaborator

@EgorkaKulikov EgorkaKulikov left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Left several comments, important moment are that direct accessor is now considered more important than setter and missed package-private fields in the lists of candidates for direct accessing.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
Archived in project
Development

Successfully merging this pull request may close these issues.

Fuzzer should change objects which are created with empty constructor by its public setters
2 participants