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

Update tests two bucket #2877

Merged
merged 18 commits into from
Jan 17, 2025
Merged

Conversation

jagdish-15
Copy link
Contributor

Pull Request

This PR adds 3 new tests to the Two Bucket exercise to sync with the problem-specifications repository. Two of the newly added tests check for error handling, which was not tested before. I am also unsure about the error message—should I keep it as it is in the problem-specifications repo or modify it to be more informative?

Let me know your thoughts on this.


Reviewer Resources:

Track Policies

@kahgoh
Copy link
Member

kahgoh commented Dec 26, 2024

I think with the error messages I think it is fine to leave them out of the exception. But, I think we should make our own exception instead of reusing IllegalArgumentException. IllegalArgumentException is used to indicate when one of the arguments to the method is bad (like if if one of them is negative). It doesn't suit cases where the arguments are good, but we can't reached the required amount. If we can't reach the value, I'd use a different exception, perhaps making a custom one for the exercise.

@jagdish-15
Copy link
Contributor Author

jagdish-15 commented Dec 26, 2024

@kahgoh, thank you for the suggestion—it does make a lot of sense to create a custom exception for this purpose.
I imagine it would look something like this:

public class UnreachableGoalException extends RuntimeException {
    public UnreachableGoalException() {
        super();
    }

    public UnreachableGoalException(String message) {
        super(message);
    }

    public UnreachableGoalException(String message, Throwable cause) {
        super(message, cause);
    }
}

That said, when I think about this, we've used IllegalArgumentException in similar scenarios before—like when the target amount couldn't be reached with the available denominations in the "Change" exercise. I've noticed this approach across many other exercises as well. (But some exercises like "Calculator Conundrum" do use custom exceptions)

While your suggestion is undoubtedly a more "proper" and robust approach, it might feel a little inconsistent with the conventions used elsewhere. What are your thoughts on this? How do you think I should proceed?

@kahgoh
Copy link
Member

kahgoh commented Dec 28, 2024

I think there is also precedence for making our own exceptions for our exercises, like BinarySearch and TreeBuilding. But yeah, the think the exception would look something like that.

@jagdish-15
Copy link
Contributor Author

Sure, I'll get right on it as soon as possible, since I already have the body of the custom Exception ready. However, I'll implement it the way it's done in exercises like "Binary Search" and "Tree Building," instead of following the approach in "Calculator Conundrum," as that causes build failure issues for Windows users (as highlighted in the forum discussion).

I was also considering implementing similar custom Exceptions in exercises like "Change," since an unassignable change value shouldn’t technically throw an IllegalArgumentException. If you have other exercises in mind that would benefit from custom Exceptions, feel free to let me know. We could even extend this improvement across the entire track—I’d be more than happy to take on that task!

@kahgoh
Copy link
Member

kahgoh commented Dec 30, 2024

However, I'll implement it the way it's done in exercises like "Binary Search" and "Tree Building," instead of following the approach in "Calculator Conundrum," as that causes build failure issues for Windows users (as highlighted in the forum discussion).

Sounds good. Feel free to update "Calculator Conundrum" as well.

I was also considering implementing similar custom Exceptions in exercises like "Change," since an unassignable change value shouldn’t technically throw an IllegalArgumentException

I'm reluctant to make this change in the other exercises. On one hand, I agree that there are other exercises where a custom exception makes sense. But on the other hand, wouldn't it mean solutions that students have already submitted become invalid if they use the custom exception? A lot of the community solutions would become invalid or outdated. The change would likely require the system to re-run tests on existing solutions, which incurs a cost (if you're interested, there is a note about expensive operations in Avoiding triggering unnecessary test runs). I'm not quite sure if the benefit is worth the cost from making the change.

@jagdish-15
Copy link
Contributor Author

jagdish-15 commented Dec 30, 2024

Sounds good. Feel free to update "Calculator Conundrum" as well.

Great! I'll get to updating "Calculator Conundrum" and this exercise as soon as possible—sorry for the delay!

Regarding the change to custom exceptions in other exercises, I completely agree with your point about the cost and the potential impact on existing solutions. Updating all exercises to use custom exceptions would indeed risk invalidating many community solutions, requiring a system-wide test rerun, which can be resource-intensive.

For this particular exercise, however, the situation seems different. Since the tests have been updated from the problem-specification repository, community solutions for this exercise are already going to be invalidated. If updates from the problem-specification repo are frequent and similar changes would cause solutions to become outdated anyway, we could consider making this adjustment now.

That said, it really comes down to the priority and whether the benefits of introducing custom exceptions outweigh the potential disruption. Let me know what you decide!

@kahgoh
Copy link
Member

kahgoh commented Dec 31, 2024

For this particular exercise, however, the situation seems different. Since the tests have been updated from the problem-specification repository, community solutions for this exercise are already going to be invalidated. If updates from the problem-specification repo are frequent and similar changes would cause solutions to become outdated anyway, we could consider making this adjustment now.

I agree, I think updating the Two Buckets exercise to meet the problem specifications and using the custom exception is fine. Its only updating the other exercises to use a custom exception like "Change" I'm reluctant to if they currently "work" with IllegalArgumentException.

1 similar comment
@kahgoh
Copy link
Member

kahgoh commented Dec 31, 2024

For this particular exercise, however, the situation seems different. Since the tests have been updated from the problem-specification repository, community solutions for this exercise are already going to be invalidated. If updates from the problem-specification repo are frequent and similar changes would cause solutions to become outdated anyway, we could consider making this adjustment now.

I agree, I think updating the Two Buckets exercise to meet the problem specifications and using the custom exception is fine. Its only updating the other exercises to use a custom exception like "Change" I'm reluctant to if they currently "work" with IllegalArgumentException.


TwoBucket twoBucket = new TwoBucket(6, 15, 5, "one");

assertThatExceptionOfType(IllegalArgumentException.class)
Copy link
Member

Choose a reason for hiding this comment

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

I think it is a bit odd to have to have to assert that we get an exception three times in the test (I was expecting only one assertion). What do you think about refactoring to assert the exception just once? In the csharp implementation, the TwoBucket class has a measure method that the tests calls. It returns a "Result" object containing the number of moves, the final bucket and the other bucket or throws an exception if the goal can be reached.

Copy link
Contributor Author

@jagdish-15 jagdish-15 Dec 31, 2024

Choose a reason for hiding this comment

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

I think it is a bit odd to have to have to assert that we get an exception three times in the test

Even I found it a bit odd to assert the exception three times while implementing the tests but couldn’t think of a better alternative. However, your suggestion of returning a Result object and handling everything within a single method is an excellent idea! This approach not only simplifies the implementation but also ensures the custom exception can be thrown effectively. Should I go ahead with this approach?

Copy link
Member

Choose a reason for hiding this comment

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

Yeah, I think we can go ahead with this one as we both seem to agree it would be better this way.

Copy link
Member

Choose a reason for hiding this comment

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

Yeah, I think we can go ahead with this one as we both seem to agree it would be better this way.

@jagdish-15
Copy link
Contributor Author

@kahgoh,

I've updated the solution so the main class now returns a Result object instead of using three separate methods to retrieve individual variables. Additionally, the constructor now throws an exception for impossible bucket configurations, which I believe is a more intuitive approach compared to delegating this responsibility to the getResult method.

I'm currently working on implementing the custom exception as we discussed. Let me know your thoughts!

@jagdish-15
Copy link
Contributor Author

@kahgoh,

I’ve updated the solution to include the custom exception as discussed. The constructor now throws this exception for impossible bucket configurations.


assertThatExceptionOfType(UnreachableGoalException.class)
.isThrownBy(() -> new TwoBucket(6, 15, 5, "one"))
.withMessage("impossible");
Copy link
Member

Choose a reason for hiding this comment

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

Now that we have the UnreachableGoalException, I think it would be enough to assert on the exception type.

Suggested change
.withMessage("impossible");

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I included the error message initially because the problem-specifications repository specified it, but I can update it if you'd prefer. Let me know what you think!

Copy link
Member

Choose a reason for hiding this comment

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

Hmm ... I think its fine to just assert the exception type here because the problem specification only describes the error as error: "impossible". Keep in mind the problem spec is written so it can be implemented in other languages too and other languages may have a different way to represent errors. For example, we might use error numbers to represent errors in C, Go has error types. We're using UnreachableGoalException to represent this type of error.

I don't mind asserting a message either, but, I'd prefer a more descriptive message that's easier to understand what went wrong (e.g. Can not reach target).

Copy link
Contributor Author

@jagdish-15 jagdish-15 Jan 16, 2025

Choose a reason for hiding this comment

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

I see your point about the cross-language nature of the problem specification. It makes sense to focus on asserting the exception type since "impossible" as a message doesn't provide much context and might not translate well to other languages or implementations.

While I agree a more descriptive message like "Cannot reach target" would be helpful, it might not be strictly necessary here. Sticking to UnreachableGoalException seems cleaner and still conveys the intent clearly. Thanks for clarifying—it feels like the right balance to avoid overcomplicating the assertion. I'll add the commit ASAP!

public void testReachingGoalIsImpossible() {

assertThatExceptionOfType(UnreachableGoalException.class)
.isThrownBy(() -> new TwoBucket(6, 15, 5, "one"))
Copy link
Member

Choose a reason for hiding this comment

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

I think we want to call the getResult() in the assertion because I think it is also valid to do calculations in the getResults method instead of the constructor.

Suggested change
.isThrownBy(() -> new TwoBucket(6, 15, 5, "one"))
.isThrownBy(() -> new TwoBucket(6, 15, 5, "one").getResult())

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I’ve updated the assertion to call getResult() as suggested. Let me know if there’s anything else you’d like adjusted!

Copy link
Member

@kahgoh kahgoh left a comment

Choose a reason for hiding this comment

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

Looks good now. Thanks @jagdish-15!

@kahgoh kahgoh merged commit 3f5c931 into exercism:main Jan 17, 2025
4 checks passed
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants