Issues with Contract Work Percentage Constraint in Employee Scheduling
Hi,
I'm working on an employee scheduling system using Timefold and have encountered an issue with implementing a contract work percentage constraint. The goal is to ensure employees are scheduled according to their contract work percentage, but I'm facing a couple of challenges:
1. Employees with 0% Contract Work Percentage:
- Currently, employees with a 0% contract work percentage are still being assigned shifts. I want to ensure they are not assigned any shifts at all.
2. Updating Contract Work Percentage:
- I'm considering updating the employee's contract work percentage dynamically based on certain conditions. Any advice on best practices for this?
Here's my current constraint implementation:
85 Replies
ā
This post has been reserved for your question.
Hey @dghf! Please useTIP: Narrow down your issue to simple and precise questions to maximize the chance that others will reply in here./close
or theClose Post
button above when your problem is solved. Please remember to follow the help guidelines. This post will be automatically marked as dormant after 300 minutes of inactivity.
Questions:
1. How can I modify the constraint to ensure employees with 0% contract work percentage are not assigned any shifts?
2. Is there a recommended way to update the employee's contract work percentage dynamically within the constraint?
Additional Context:
- I'm using Timefold 1.19.0 with Quarkus.
- Other constraints, like shift preferences, are working fine.
Any insights or suggestions would be greatly appreciated!
Thank you!
If you are finished with your post, please close it.
If you are not, please ignore this message.
Note that you will not be able to send further messages here after this post have been closed but you will be able to create new posts.
not sure if thats what you want but couldnt you filter if work percentage is 0 for a given employee
in the .filter() part
@ayylmao123xdd It didnt work, even when you filter, it's still assigned the employee.
Can you show your modified filtering?
@dan1st | Daniel
shouldn't that condition be inverted?
normally, the filter gives you the elements that match the condition
I got this from doing PUT
And this is the reworked code
looks about right
Does that match your expectations?
@dan1st | Daniel still Alice assigned to one, when their work percentage is 0:
Both should've been assigned to Bob
Is that the result of filtering?
My friend will enter the conversation, we're working together @dan1st | Daniel, he can explain better š
that the result of the solving if you look on the shifts that been assigned we see that Alice is assigned to first one while Alice workPercentage is 0
"shifts": [
This message has been formatted automatically. You can disable this using
/preferences
.It does say
"solverStatus": "NOT_SOLVING"
For this shift, the end is before the start
Are you using some linear programming/linear constrained optimization library?The solve status is a flag that indicates when the solving process is finishing. Since the Timefold solver performs solving while it's still in progress, the flag will show that it is still solving. Therefore, the status does not indicate anything about the shifts.
"solverStatus": "SOLVING_ACTIVE"
while its still solving
The code is using a library called Timefold Solver, which is a constraint solver. It is used for solving planning and scheduling problems by defining constraints and optimizing solutions based on those constraints. The constraints are defined in our
EmployeeSchedulingConstraintProvider
class, using the Timefold Solver's API, which includes classes like ConstraintFactory
, ConstraintCollectors
, and HardSoftBigDecimalScore
. These are used to define and manage constraints for employee scheduling, such as ensuring no overlapping shifts, respecting employee preferences, and balancing shift assignments.well in the output it was In the output, it was
NOT_SOLVING
with "feasible": false
Can I see the exact input?This message has been formatted automatically. You can disable this using
/preferences
.@dan1st | Daniel
Can you show the
Shift
and Employee
classes?i am back
also for this piece right here
shouldnt you add like
a check
where you do
current worked hours + the hours you gonna work
and check if its above max allowed hours
because if im getting it right
if you are at lets say 159 worked hours
and max allowed is 160
then you will be able to work the whole shift anyway
@dan1st | Daniel
@ayylmao123xdd so doing this?
i guess
give it a try
I think the logic is correct, but the current approach first calculates the shift's hours, then checks if the employee's work percentage allows them to take the shift. Instead, it should update the employee's remaining work percentage after assigning them a shift.
totalWorkedHours is the shift hours
no clue what that means
im too low on sunlight today
š
disaster right
.groupBy(
(employee, shift) -> employee,
ConstraintCollectors.sum ((employee, shift) ->
(int) Duration.between(shift.getStart(), shift.getEnd()).toHours())
calculate the shift hours
hmm
maybe
so is this one correct or not
if i look at it without thinking much i would say u need the total worked hours + current shift hours check
Duration.between(shift.getStart(), shift.getEnd()).toHours() represents the hours of a single shift
and totalWorkedHours should be the sum of all assigned shift hours for an employee.
then
this should be correct i think
yea so basically what i said ig
Note: 1h59min will result in you getting 1h
total worked hours + duration of single shift <= max allowed hours
free unpaid overtime
.toMinutes() and divide 60?
ye that correct
but how
Well when should it count the hour and when shouldn't it?
well
if i was the employee
1 hour and 1 minute would count as 2 hours of work
the problem is i cant access the duration of single shift in filter
maybe count the minutes then have a logic for the hours?
what do you want to do if you have like 45min?
or 30min?
currently you are using integer hours for everything so you cannot do scheduling with anything except hours
i meaaaaaaaaaaaaaaaaaaaaaaaan
if its a normal job
ur paid by hour
so
ig counting the hours is the best option
like even if its 1 hour 10 minutes count it as 2 h
most are computing it more granually
like if you have 30min on one day and 30min on another day, you have 1h in total
depends on the company ig then
so in that case
i would count the minutes
instead of hours
same for
Correct, I didn't fully think about that š
. Should we use 30-minute increments, or just round to the closest hour?
totalworkedminutes instead of hours
its a thesis work so not really a big deal
idk what you are doing - it might be best to just schedule everything based on minutes
@dan1st | Daniel But even if we count in minutes instead of hours, shouldn't workpercentage being 0 mean that no shifts are assigned to that particular employee?
The problem could lie somewhere in the logic here
(from the OP)
minutes vs hours shouldn't change anything there
Yeah, but do you know what could be the issue instead? Should we do more logging somewhere to see what could be causing the issue along the way?
Does the library allow doing custom logging?
oh wait
you are using
!=
on double
s
don't do that
double
s aren't exact values
I think return totalWorkedHours.toHours() < desiredHours
should be ok
or return totalWorkedHours.toHours() < desiredHours - 0.0001
or similarI think return totalWorkedHours.toHours() < desiredHours should be okwe've tried this but it didn't work But why
return totalWorkedHours.toHours() < desiredHours - 0.0001
You mean why the -0.0001?
Yeah
Like why do we subtract 0.0001 from
desiredHours
?If you are using
double
, you are not dealing with precise values
for example if you compute 0.1+0.2
, you'll get something like 0.2999999... and not 0.3or maybe just use bigdouble
and I'm subtracting a delta to ensure it's actually smaller
for calculations
or whatever that class was called
it's called
BigDecimal
but there are some other issues with it
tbh fixed point mathw ith long
might also work lolor just swap to minutes
š±
so u dont have 1.1 minute
ezzzzzzzzzzz
the only thing where decimal numbers are coming from here is the percentages lol
cant u just floor it
then floor(0.2999999999) gives 0.2
oh yea
i meant the round
l0l
same problem with 0.49999999
the solutions is comparing stuff with a delta
disaster then
using a delta is no problem
delta force
im rewriting the whole constriant hopes it becomes better
This message has been formatted automatically. You can disable this using
/preferences
.what do you think? is that better way?
@dan1st | Daniel
looks about fine
i subtract the shiftminutes from the employeeworkminutes "workpercentage"
then im thinking about updating the working percentage each time a shift assign to the employee
You can try logging the employee, shift and the returned value in every
filter()
callHmm I am not sure how that'd be done. Any suggestions?
Finally its working :3 thank you for your help
This message has been formatted automatically. You can disable this using
/preferences
.it loop through Shift.class instead of Employee.class because we want to check actual assignments.
then it check if the shift has an assigned employee (shift.getEmployee() != null).
if it does then we go through the same logic
@dan1st | Daniel
We'll verify that employees do not exceed their assigned work percentage. For example, if an employee has a 1% work percentage, they should only be assigned shifts that fit within that limit.
š¤
Post marked as dormant
This post has been inactive for over 300 minutes, thus, it has been archived.
If your question was not answered yet, feel free to re-open this post or create a new one.
In case your post is not getting any attention, you can try to use /help ping
.
Warning: abusing this will result in moderative actions taken against you.