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

New concept: Optional #2913

Draft
wants to merge 19 commits into
base: main
Choose a base branch
from

Conversation

josealonso
Copy link
Contributor

Write explanation for the Optional concept, as well as the test file, the stub implementation, the reference solution and other required files. Some files must be modified. This is a draft.

This solves #2555


Reviewer Resources:

Track Policies

… the stub implementation, the reference solution and other required files. Some files must be modified. This is a draft.
@josealonso
Copy link
Contributor Author

josealonso commented Feb 10, 2025

I just committed a first draft. Don't worry about the linting errors, I will sort them out during this week.
I'd like to receive feedback about the reference solution and the test file, the bulk of the work is done.

  • Is there an exercise for the generics concept ? That is a prerequisite for this concept.
  • If that concept is not explained on the Java track, I can adapt it from other track.

@kahgoh @jagdish-15

@kahgoh
Copy link
Member

kahgoh commented Feb 11, 2025

Thanks @josealonso! I plan to have a look at the draft soon.

In regards to the generics concept, it is covered as generics type.

@josealonso
Copy link
Contributor Author

Thanks @josealonso! I plan to have a look at the draft soon.

In regards to the generics concept, it is covered as generics type.

Thanks for the link to the generic-types exercise!! Shame on me

@josealonso
Copy link
Contributor Author

Which icon should I choose?

@kahgoh
Copy link
Member

kahgoh commented Feb 12, 2025

Did you mean the icon for the exercise? They get added to exercism/icons repository, though that can be done after the exercise is added. We'll have to make one, but I think it will be the same icon as the original Tim From Marketing, but with a "2" in the top right corner. Compare Wizards and Warriors and Wizards and Warriors 2 for an example.



class EmployeeService {

Copy link
Member

Choose a reason for hiding this comment

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

To be honest, I didn't quite get the purpose of these two methods (getAllTheEmployeesById and getEmployeeById) especially as we're not expecting the students to implement these and I don't think its clear when we expect them to call them.

Since the Employee have methods that return Optional objects, I'd suggest having the test pass Employee objects or a List of them instead.

Is it also worth having a task for where the student has to return an Optional as well?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Thanks for your feedback.
I agree, I did it like that because I had all the code in one single file.

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 think this has been resolved.

Copy link
Member

@kahgoh kahgoh Feb 15, 2025

Choose a reason for hiding this comment

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

I had another thought - the getEmployeeById could also be left up to the student to implement. We already have the foreach concept, so they could use that to find the employee and that'll give them the chance to practice making an Optional object (assuming they are looking through a List<Employee>) and there would be no need for streams in the class.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

@kahgoh, despite I spent many hours creating this exercise, I have to admit it's probably a bit complex for a learning exercise. This is what I propose:

  • Leaving this one for a future practice exercise.
  • Create a concept exercise for Streams first. And maybe another one for Suppliers or functional interfaces, based on my article or a similar one.

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'll leave a longer answer tomorrow, but I just wanted to thank you for your support and clarify that my struggle is due to being my first exercise designed from scratch and also because I wasn't not very familiar with the Optional API.
I would cover more simple streams operations in a first Streams learning exercise, like map, filter and other simple operations.
@kahgoh

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 had another thought - the getEmployeeById could also be left up to the student to implement. We already have the foreach concept, so they could use that to find the employee and that'll give them the chance to practice making an Optional object (assuming they are looking through a List<Employee>) and there would be no need for streams in the class.

Are you saying I can implement the same methods without using streams ? I'm not sure that's doable. 🤔

Copy link
Member

@kahgoh kahgoh Feb 20, 2025

Choose a reason for hiding this comment

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

I was thinking the EmployService could take in a List<Employee> in the constructor, so then the students could be tasked with finding the Employee by id. I imagine the stub would like this:

class EmployeeService {
  public EmployeeService(List<Employee> employees) {
    throw new UnsupportedOperationException("Please implement the EmployeeService constructor");
  }

  public Optional<Employee> getEmployeeById(int id) {
     throw new UnsupportedOperationException("Please implement the EmployeeService.getEmployeeById() method");
  }
}

Students can still make a solution for this using a for / while loop that goes through the list until it finds the Employee. Here's one rough way:

for (Employee candidate: employeesList) {
  if (Objects.equals(candidate.getId(), Optional.of(id)) {
    return Optional.of(candidate);
  }
}

return Optional.empty();

Alternatively, I think passing the method a List<Employee> to the method would also work.

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 was thinking the EmployService could take in a List<Employee> in the constructor, so then the students could be tasked with finding the Employee by id. I imagine the stub would like this:

class EmployeeService {
  public EmployeeService(List<Employee> employees) {
    throw new UnsupportedOperationException("Please implement the EmployeeService constructor");
  }

  public Optional<Employee> getEmployeeById(int id) {
     throw new UnsupportedOperationException("Please implement the EmployeeService.getEmployeeById() method");
  }
}

Students can still make a solution for this using a for / while loop that goes through the list until it finds the Employee. Here's one rough way:

for (Employee candidate: employeesList) {
  if (Objects.equals(candidate.getId(), Optional.of(id)) {
    return Optional.of(candidate);
  }
}

return Optional.empty();

Alternatively, I think passing the method a List<Employee> to the method would also work.

Thanks @kahgoh. I'll try to change it this weekend. I was stuck, because I designed this exercise to try to mimic the access to a database. But you're right, I made it too complex.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Thanks for your help, @kahgoh. Take a look when you have a chance. I think I did it the way you suggested. I'm sure it can be improved, but streams are not used in the reference solution.

@josealonso
Copy link
Contributor Author

Did you mean the icon for the exercise? They get added to exercism/icons repository, though that can be done after the exercise is added. We'll have to make one, but I think it will be the same icon as the original Tim From Marketing, but with a "2" in the top right corner. Compare Wizards and Warriors and Wizards and Warriors 2 for an example.

Who can make that icon? I'm sorry I don't know how to edit a simple icon :(

…ns have been added. The tests have been corrected and integrated in an IDE. What remains are the hints.md and the icon files, and the gradle files for this exercise.
@josealonso
Copy link
Contributor Author

I don't understand why the java-test-runner failed, @kahgoh. It probably has to do with some missing configuration files,

@kahgoh
Copy link
Member

kahgoh commented Feb 15, 2025

Who can make that icon? I'm sorry I don't know how to edit a simple icon :(

That's fine, we can do that later. For now, let's just work on the concept content.

@kahgoh
Copy link
Member

kahgoh commented Feb 15, 2025

I don't understand why the java-test-runner failed, @kahgoh. It probably has to do with some missing configuration files,

Its a bit hard to find, but this appears in the log:

tim-from-marketing-2: exemplar solution did not pass the tests

Are the tests passing locally for you?

@josealonso
Copy link
Contributor Author

Are the tests passing locally for you?

Yes.

@manumafe98 manumafe98 added the x:size/large Large amount of work label Feb 17, 2025
@manumafe98
Copy link
Contributor

Maybe this doc on configlet could be useful for you as well to check

@josealonso
Copy link
Contributor Author

Maybe this doc on configlet could be useful for you as well to check

Thanks, I went through it already. I'll have to read it again.

@josealonso
Copy link
Contributor Author

@kahgoh, can you please go through these files and tell me if that's the right approach ? Thanks :)

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.

Hey @josealonso, I have had a look through and I think the exercise is going in the right direction.

In a couple of the comments I mention an about.md. The about.md is shown after they complete the exercise and would be something that will need to eventually be added for the concept.


@BeforeEach
void setup() {
initList();
Copy link
Member

Choose a reason for hiding this comment

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

I found this style of setting up the list (declare the list as field, call initList to populate the list and set up EmployeeService with it here) a little hard to follow because there is quite a bit of jumping around to figure out who the employees are. I'd suggest just setting up the list here:

Suggested change
initList();
listOfEmployees = List.of(
new Employee(1, "Tim", "Marketing"),
new Employee(2, "Joe", "Sales"),
new Employee(3, "Jane", "IT"),
new Employee(4, null, null)
);

Copy link
Contributor Author

@josealonso josealonso Mar 11, 2025

Choose a reason for hiding this comment

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

I couldn't do it like that, because I was mutating a list element in the null case. I will revisit it.

Copy link
Member

Choose a reason for hiding this comment

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

I'm not quite sure where you are mutating the list element, but if a test needs different data, I'd suggest setting up the test data in the test methods instead.

listOfEmployees.add(new Employee(1, "Tim", "Marketing"));
listOfEmployees.add(new Employee(2, "Joe", "Sales"));
listOfEmployees.add(new Employee(3, "Jane", "IT"));
listOfEmployees.add(new Employee(4, null, null));
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 there are a number of cases missing, such as a null name and non-null department and non-null name and null department.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

WIP

@josealonso josealonso force-pushed the 2555-new-concept-optional branch from 2168b21 to 53cbbda Compare March 14, 2025 01:44
@josealonso
Copy link
Contributor Author

I will continue tomorrow.

@kahgoh
Copy link
Member

kahgoh commented Mar 14, 2025

I will continue tomorrow.

No worries @josealonso . There's no hurry for this one, so feel free to take longer if you need to

@josealonso
Copy link
Contributor Author

josealonso commented Mar 14, 2025

I will continue tomorrow.

No worries @josealonso . There's no hurry for this one, so feel free to take longer if you need to

  • The value of concepts.uuid is a zero-length string.
    I thought it had to match the exercises.uuid, but doing that produced an error, so I left it empty.
    It's the only error in the configlet phase.

  • The tests are failing to build even locally, it might be a configuration issue.

I addressed all the suggestions you made, except using List.of() instead of a function in the tests.

Please tell me how to fix the two errors mentioned above and fell free to suggest any improvements.

Thanks @kahgoh!

@kahgoh
Copy link
Member

kahgoh commented Mar 19, 2025

The value of concepts.uuid is a zero-length string.
I thought it had to match the exercises.uuid, but doing that produced an error, so I left it empty.

They are two different UUIDs. You can generate one using configlet uuid.

The tests are failing to build even locally, it might be a configuration issue.

I'm a little confused as to which solution to look at as there is one in src/reference/java and I can also see an implementation in src/main/java. I assume the one in src/main/java is just a work in progress (ultimately, we want to end up with the working one in src/referece/java). Could you let me know how you are currently testing?

Btw, to test against the reference solution, you can go into the exercises directory and run ./gradlew :concept:tim-from-marketing-2:test.

@kahgoh
Copy link
Member

kahgoh commented Mar 19, 2025

I've just tried running the tests locally. The failing tests have the following messages:

EmployeeDatabaseTest > Retrieve employee details by id when department is null and name is not null FAILED
    org.opentest4j.AssertionFailedError:
    expected: "No employee found for id: 6"
     but was: "6 - Alice - null"
        at [email protected]/jdk.internal.reflect.DirectConstructorHandleAccessor.newInstance(DirectConstructorHandleAccessor.java:62)
        at [email protected]/java.lang.reflect.Constructor.newInstanceWithCaller(Constructor.java:502)
        at app//EmployeeDatabaseTest.retrieveEmployeeDetailsById_whenDepartmentIsNull(EmployeeDatabaseTest.java:92)

and

EmployeeDatabaseTest > Retrieve employee details by id when the id does not exist FAILED
    org.opentest4j.AssertionFailedError:
    expected: "No employee found for id: 10"
     but was: "That id does not exist: 10"
        at [email protected]/jdk.internal.reflect.DirectConstructorHandleAccessor.newInstance(DirectConstructorHandleAccessor.java:62)
        at [email protected]/java.lang.reflect.Constructor.newInstanceWithCaller(Constructor.java:502)
        at app//EmployeeDatabaseTest.retrieveEmployeeDetailsById_forANonExistingId(EmployeeDatabaseTest.java:100)

I ran them against the .meta/src/reference using gradlew :concept:tim-from-marketing-2:test. Can you see why these test may fail?

@josealonso
Copy link
Contributor Author

The value of concepts.uuid is a zero-length string.
I thought it had to match the exercises.uuid, but doing that produced an error, so I left it empty.

They are two different UUIDs. You can generate one using configlet uuid.

The tests are failing to build even locally, it might be a configuration issue.

I'm a little confused as to which solution to look at as there is one in src/reference/java and I can also see an implementation in src/main/java. I assume the one in src/main/java is just a work in progress (ultimately, we want to end up with the working one in src/referece/java). Could you let me know how you are currently testing?
Oh, sorry, I thought both files had the same content. I changed only the file name, as you suggested. I might have changed something else without realizing it. Let me take a look tonight (in Europe).
Thanks for the hint, because I couldn't run the tests locally.

Btw, to test against the reference solution, you can go into the exercises directory and run ./gradlew :concept:tim-from-marketing-2:test.
Thanks, hopefully this command will work locally!

@josealonso
Copy link
Contributor Author

I've just tried running the tests locally. The failing tests have the following messages:

EmployeeDatabaseTest > Retrieve employee details by id when department is null and name is not null FAILED
    org.opentest4j.AssertionFailedError:
    expected: "No employee found for id: 6"
     but was: "6 - Alice - null"
        at [email protected]/jdk.internal.reflect.DirectConstructorHandleAccessor.newInstance(DirectConstructorHandleAccessor.java:62)
        at [email protected]/java.lang.reflect.Constructor.newInstanceWithCaller(Constructor.java:502)
        at app//EmployeeDatabaseTest.retrieveEmployeeDetailsById_whenDepartmentIsNull(EmployeeDatabaseTest.java:92)

and

EmployeeDatabaseTest > Retrieve employee details by id when the id does not exist FAILED
    org.opentest4j.AssertionFailedError:
    expected: "No employee found for id: 10"
     but was: "That id does not exist: 10"
        at [email protected]/jdk.internal.reflect.DirectConstructorHandleAccessor.newInstance(DirectConstructorHandleAccessor.java:62)
        at [email protected]/java.lang.reflect.Constructor.newInstanceWithCaller(Constructor.java:502)
        at app//EmployeeDatabaseTest.retrieveEmployeeDetailsById_forANonExistingId(EmployeeDatabaseTest.java:100)

I ran them against the .meta/src/reference using gradlew :concept:tim-from-marketing-2:test. Can you see why these test may fail?

I'll let you know if I can run the tests locally. Thanks!


```java
Employee nullableEmployee = new Employee();
Optional<Employee> nullableEmployee = Optional.ofNullable(employee);
Copy link
Member

Choose a reason for hiding this comment

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

nullableEmployee is never null here. I think a better example would be to call a method that might return null.

I'd also suggest Optional.empty.

If the employee _may_ be not present, the static [ofNullable][optional-ofNullable-javadoc] method must be used:

```java
Employee nullableEmployee = new Employee();
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 a better example would be calling a method that might return null (in this example, it is never null). I'd also suggest calling the variable optionalEmployee because nullable tends mean you can assign a null value to it.


## Basic methods

If a value is present, the [isPresent][optional-isPresent-javadoc] method returns true and the [get][optional-get-javadoc] method returns the value.
Copy link
Member

Choose a reason for hiding this comment

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

We've switch from "may return a value" or "no value" from the introduction to "present" here. I think it would help to define what "present" and "empty" mean either here or back in the introduction.

}

@Test
@Tag("task:3")
Copy link
Member

Choose a reason for hiding this comment

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

There are only two tasks in the instructions, so the tags should be only "task:1" or "task:2". They should line up with the corresponding task in the instructions.


@BeforeEach
void setup() {
initList();
Copy link
Member

Choose a reason for hiding this comment

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

I'm not quite sure where you are mutating the list element, but if a test needs different data, I'd suggest setting up the test data in the test methods instead.

@@ -0,0 +1,2 @@
[]
Copy link
Member

Choose a reason for hiding this comment

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

Suggest a link to the Javadocs for Optional. See the links.json for the Lists concept for an example.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
x:size/large Large amount of work
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants