I'm learning OR-Tools and started adding onto the nurse scheduling problem. I'm trying to add a constraint that if a nurse is working that they can only work consecutive shifts (i.e can't work shifts 1 and 3, but can work shift 2 and 3).
I'm not entirely certain what is wrong with my code. When the nurses have to ability to work at least (num_of_shifts - 1) shifts per day it works, but if they cannot it does not (e.g. works if there are 5 shifts in a day each nurse can work 4 shifts in a day).
Here is an example output, there are 3 shifts, 3 nurses and every nurse can work a max of 2 shifts (ignore the not requested part, I got rid of the code for it):
Day 0
Nurse 0 works shift 1 (not requested).
Nurse 0 works shift 2 (not requested).
Nurse 1 works shift 0 (not requested).
Nurse 1 works shift 1 (not requested).
Nurse 2 works shift 0 (not requested).
Day 1
Nurse 0 works shift 1 (not requested).
Nurse 0 works shift 2 (not requested).
Nurse 2 works shift 0 (not requested).
Day 2
Nurse 0 works shift 1 (not requested).
Nurse 0 works shift 2 (not requested).
Nurse 2 works shift 0 (not requested).
Nurse 2 works shift 1 (not requested).
Day 3
Nurse 0 works shift 0 (not requested).
Nurse 0 works shift 1 (not requested).
Nurse 1 works shift 1 (not requested).
Nurse 1 works shift 2 (not requested).
Day 4
Nurse 0 works shift 0 (not requested).
Nurse 1 works shift 1 (not requested).
Nurse 1 works shift 2 (not requested).
Nurse 2 works shift 1 (not requested).
This is the code that isn't working
has_started_shifts = {}
for n in all_nurses:
for d in all_days:
has_started_shifts[(n, d)] = model.NewBoolVar(f"has_started_shifts_n{n}_d{d}")
# consecutive shifts constraint
for n in all_nurses:
for d in all_days:
# If a nurse works a shift, mark that they have started shifts for the day
for s in all_shifts:
model.Add(has_started_shifts[(n, d)] == True).OnlyEnforceIf(shifts[(n, d, s)])
# If a nurse works a shift, they can work any subsequent shift
if s > 1:
model.Add(shifts[(n, d, s)] <= shifts[(n, d, s - 1)]).OnlyEnforceIf(has_started_shifts[(n, d)])
I have tried creating auxiliary variable that keep track of if the first shift has happened yet but that didn't work either.
I have studied https://github.com/google/or-tools/blob/stable/examples/python/shift_scheduling_sat.py but it doesn't implement what I want.
Thanks for any help you can provide.
I figured a solution out. The way I fixed it was by finding the first and last shift for a person, then subtracting those values to find the difference, then making a constraint that a nurse must work exactly the difference in the amount of shifts for the day