Task description
There is a queue for the self-checkout tills at the supermarket. Your task is write a function to calculate the total time required for all the customers to check out!
Input:
customers: An integer array representing customer estimated processing times.
tillCount: An integer representing available tills.output:
An integer representing the maximum time required to process all customers.Examples:
queueTime([5,3,4], 0)
returns12
queueTime([10,2,3,3], 2)
returns10
queueTime([2,3,10], 2)
returns12
Task solution
Tests
For this Kata I have chosen to implement the functionality in JavaScript, this being the case, I will use jest as the test runner for our test cases.
We have need to test the following failure cases:
- If the
customers
parameter is not an array - If the
customers
parameter is an array containing non-integer types - If the
tillCount
parameter is not an integer
We then continue on to implement our happy path cases:
- If noone is in line, no wait time should be expected
- If customers are in line, total their wait times based on the tills available
describe("example tests", () => {
it("Should throw if invalid inputs provided", () => {
expect(() => queueTime(1, 1)).toThrow(/InvalidArgumentException/);
expect(() => queueTime(["test", 2, null], 1)).toThrow(/InvalidArgumentException/);
expect(() => queueTime([], null)).toThrow(/InvalidArgumentException/);
});
it("Should have no queue time if no customers are in line", () => {
expect(queueTime([], 1)).toBe(0);
});
it("Should calculate the correct queue time for valid customers", () => {
expect(queueTime([5,3,4], 0)).toBe(12);
expect(queueTime([1,2,3,4], 1)).toBe(10);
expect(queueTime([2,2,3,3,4,4], 2)).toBe(9);
expect(queueTime([1,2,3,4,5], 100)).toBe(5);
});
});
Implementation
function queueTime(customers, tillCount) {
if(!Array.isArray(customers)) {
throw new Error(`InvalidArgumentException: Parameter 1 must be an array, received: ${typeof customers}`);
} else if(!customers.every(time => Number.isInteger(time))) {
throw new Error(`InvalidArgumentException: Parameter 1 must be an array of integers. Atleast one element in the array does not conform to this, received: ${customers}`);
} else if(!Number.isInteger(tillCount)) {
throw new Error(`InvalidArgumentException: Parameter 2 must be an integer, received: ${typeof tillCount}`);
}
let tills = Array(tillCount <= 0 ? 1 : tillCount).fill(0);
customers.forEach(customer => {
const fastest = tills.indexOf(Math.min(...tills));
tills[fastest] += customer;
});
return Math.max(...tills);
}
We begin by running our checks as usual. Then we then begin to hit an edge case just as we begin our happy path implementation, consider this:
The
tillCount
is0
but the customer wait times are existing and valid. This being the case, we are to assume based on the task description that alternative arrangements have been made to account for these customers such as a person manually doing the work π€·ββοΈ. This being the case, the wait times should be processed as if1
till is actually active, thus, atillCount
of0
resolves the same result as atillCount
of1
.
This being the case, we check if tillsCount
is 0
or less and if it is, we assume it to be equivelant to 1
, otherwise we use whatever tillsCount
is actually set to. We also have this case covered in our TDD flow on this line:
expect(queueTime([5,3,4], 0)).toBe(12);
The reason for this is simple, if we were to set new Array(0).fill(0)
, we would get a -Infinity
value returned every time from the queueTime
function. The reason for that is quite silly but also kind of makes sense. Basically, the array, if it had been created as new Array(0)
would have no elements and thus the .fill(0)
populates no indicies since none exist, it's an empty array afterall. From here as we run the loop for getting our customer times validated. At this point, the indexOf
call returns -1
since no index is found, since none exist. So far things make sense but here is where it gets silly. As we execute the tills[fastest] += customer;
line, JavaScript will allow us to set an index of -1
on the array and assign it a value, in this case, NaN
. Thus, our tills
array after the loop is finished will always be [-1: NaN]
. You might rightfully be thinking "how is that even a thing?", well it gets slightly worse because in Javascript, an array with negative indexes is invalid and thus, when we call Math.max(...tills);
JavaScript interperets that as Math.max(...[])
and the default return value in such cases of using Math.max
is -Infinity
. Before you ask, the flipside case of using Math.min
will return Infinity
under the same conditions, so atleast there's a predictable and consistent implementation π.
So, understanding these quirks, we move onto the loop itself which simple checks what the till with the lowest wait time is and adds the current customer in the loop to it. Lets imagine the following pseudo-code:
customers: [1, 2, 3]
tills: [0, 0]
loop customers
1st iteration -> tills = [1, 0]
2nd iteration -> tills = [1, 2]
3rd iteration -> tills = [4, 2]
Maximum wait time -> 4
This being the case, we simply return the maximum value in the tills array to finish things.
Conclusions
This was quite a fun Kata to work with, I remember completing it a couple of weeks back and finding the quirks with Math.min
and Math.max
that I hadn't come across in almost 8 years of development with JavaScript but it is one of these things you come across and you just think to yourself "that's pretty cool but also... why?". I guess that is one of the reasons JavaScript continues to be such a popular language, it is powerful in and of itself but it is so quirky that you learn something new almost every day π.
I experimented with using a reducer as the final return value like so:
// code removed to keep this snippet short
let tills = Array(tillCount === 0 ? 1 : tillCount).fill(0);
return customers.reduce((maxTime, customer) => {
const fastest = tills.indexOf(Math.min(...tills));
tills[fastest] += customer;
return Math.max(...tills);
}, 0);
This works just the same as the implementation above but personally, I don't like the use of tills
inside the reducer function since it is not explicitely passed in. Perhaps this is just me but either way I settled on the implementation we went over in the section above and I am pretty happy with the outcome.
Top comments (0)