Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions bin/auto-sync.txt
Original file line number Diff line number Diff line change
Expand Up @@ -82,6 +82,7 @@ spiral-matrix
state-of-tic-tac-toe
strain
sublist
swift-scheduling
twelve-days
two-bucket
two-fer
Expand Down
8 changes: 8 additions & 0 deletions config.json
Original file line number Diff line number Diff line change
Expand Up @@ -489,6 +489,14 @@
"floating_point_numbers"
]
},
{
"slug": "swift-scheduling",
"name": "Swift Scheduling",
"uuid": "d080d214-1d41-4347-a494-1e05db9f614a",
"practices": [],
"prerequisites": [],
"difficulty": 1
},
{
"slug": "transpose",
"name": "Transpose",
Expand Down
50 changes: 50 additions & 0 deletions exercises/practice/swift-scheduling/.docs/instructions.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
# Instructions

Your task is to convert delivery date descriptions to _actual_ delivery dates, based on when the meeting started.

There are two types of delivery date descriptions:

1. Fixed: a predefined set of words.
2. Variable: words that have a variable component, but follow a predefined set of patterns.

## Fixed delivery date descriptions

There are three fixed delivery date descriptions:

- `"NOW"`
- `"ASAP"` (As Soon As Possible)
- `"EOW"` (End Of Week)

The following table shows how to translate them:

| Description | Meeting start | Delivery date |
| ----------- | ----------------------------- | ----------------------------------- |
| `"NOW"` | - | Two hours after the meeting started |
| `"ASAP"` | Before 13:00 | Today at 17:00 |
| `"ASAP"` | After or at 13:00 | Tomorrow at 13:00 |
| `"EOW"` | Monday, Tuesday, or Wednesday | Friday at 17:00 |
| `"EOW"` | Thursday or Friday | Sunday at 20:00 |

## Variable delivery date descriptions

There are two variable delivery date description patterns:

- `"<N>M"` (N-th month)
- `"Q<N>"` (N-th quarter)

| Description | Meeting start | Delivery date |
| ----------- | ------------------------- | --------------------------------------------------------- |
| `"<N>M"` | Before N-th month | At 8:00 on the _first_ workday of this year's N-th month |
| `"<N>M"` | After or in N-th month | At 8:00 on the _first_ workday of next year's N-th month |
| `"Q<N>"` | Before or in N-th quarter | At 8:00 on the _last_ workday of this year's N-th quarter |
| `"Q<N>"` | After N-th quarter | At 8:00 on the _last_ workday of next year's N-th quarter |

~~~~exercism/note
A workday is a Monday, Tuesday, Wednesday, Thursday, or Friday.

A year has four quarters, each with three months:
1. January/February/March
2. April/May/June
3. July/August/September
4. October/November/December.
~~~~
6 changes: 6 additions & 0 deletions exercises/practice/swift-scheduling/.docs/introduction.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
# Introduction

This week, it is your turn to take notes in the department's planning meeting.
In this meeting, your boss will set delivery dates for all open work items.
Annoyingly, instead of specifying the _actual_ delivery dates, your boss will only _describe them_ in an abbreviated format.
As many of your colleagues won't be familiar with this corporate lingo, you'll need to convert these delivery date descriptions to actual delivery dates.
19 changes: 19 additions & 0 deletions exercises/practice/swift-scheduling/.meta/config.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
{
"authors": [
"resu-xuniL"
],
"files": {
"solution": [
"SwiftScheduling.php"
],
"test": [
"SwiftSchedulingTest.php"
],
"example": [
".meta/example.php"
]
},
"blurb": "Convert delivery date descriptions to actual delivery dates.",
"source": "Erik Schierboom",
"source_url": "https://github.com/exercism/problem-specifications/pull/2536"
}
104 changes: 104 additions & 0 deletions exercises/practice/swift-scheduling/.meta/example.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,104 @@
<?php

declare(strict_types=1);

class SwiftScheduling
{
private const QUARTER = [
1 => 4,
2 => 7,
3 => 10,
4 => 13
];

public function deliveryDate(DateTime $meetingStart, string $description): ?DateTime
{
switch ($description) {
case 'NOW':
return $this->convertNow($meetingStart);
case 'ASAP':
return $this->convertAsap($meetingStart);
case 'EOW':
return $this->convertEow($meetingStart);
default:
if (str_ends_with($description, "M")) {
return $this->convertVariableM($meetingStart, (int) substr($description, 0, -1));
} elseif (str_starts_with($description, "Q")) {
return $this->convertVariableQ($meetingStart, (int) substr($description, 1,));
} else {
return null;
}
}
}

private function convertNow(DateTime $meetingStart): DateTime
{
return $meetingStart->modify('+2 hours');
}

private function convertAsap(DateTime $meetingStart): DateTime
{
$meetingHour = (int) $meetingStart->format('H');

return $meetingHour < 13
? $meetingStart->setTime(17, 0)
: $meetingStart->setTime(13, 0)
->modify('+1 day');
}

private function convertEow(DateTime $meetingStart): DateTime
{
$meetingDay = (int) $meetingStart->format('w');

return $meetingDay === 1 || $meetingDay === 2 || $meetingDay === 3
? $meetingStart->modify("friday")
->setTime(17, 0)
: $meetingStart->modify("sunday")
->setTime(20, 0);
}

private function convertVariableM(DateTime $meetingStart, int $dueMonth): DateTime
{
$meetingMonth = (int) $meetingStart->format('m');

$meetingMonth < $dueMonth
? $meetingStart->setDate(
(int) $meetingStart->format('Y'),
$dueMonth,
1
)
: $meetingStart->setDate(
(int) $meetingStart->format('Y') + 1,
$dueMonth,
1
);

$meetingDay = (int) $meetingStart->format('w');

return $meetingDay === 0 || $meetingDay === 6
? $meetingStart->modify("first weekday")
->setTime(8, 0)
: $meetingStart->setTime(8, 0);
}

private function convertVariableQ(DateTime $meetingStart, int $dueQuarter): DateTime
{
$meetingMonth = (int) $meetingStart->format('m');
$meetingQuarter = round(($meetingMonth + 2) / 3);

$meetingStart->setDate(
(int) $meetingStart->format('Y'),
self::QUARTER[$dueQuarter],
1
);

if ($meetingQuarter > $dueQuarter) {
$meetingStart->modify('+1 year');
}

$meetingStart->modify("last weekday")
->setTime(8, 0);

return $meetingStart;
}
}
58 changes: 58 additions & 0 deletions exercises/practice/swift-scheduling/.meta/tests.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
# This is an auto-generated file.
#
# Regenerating this file via `configlet sync` will:
# - Recreate every `description` key/value pair
# - Recreate every `reimplements` key/value pair, where they exist in problem-specifications
# - Remove any `include = true` key/value pair (an omitted `include` key implies inclusion)
# - Preserve any other key/value pair
#
# As user-added comments (using the # character) will be removed when this file
# is regenerated, comments can be added via a `comment` key.

[1d0e6e72-f370-408c-bc64-5dafa9c6da73]
description = "NOW translates to two hours later"

[93325e7b-677d-4d96-b017-2582af879dc2]
description = "ASAP before one in the afternoon translates to today at five in the afternoon"

[cb4252a3-c4c1-41f6-8b8c-e7269733cef8]
description = "ASAP at one in the afternoon translates to tomorrow at one in the afternoon"

[6fddc1ea-2fe9-4c60-81f7-9220d2f45537]
description = "ASAP after one in the afternoon translates to tomorrow at one in the afternoon"

[25f46bf9-6d2a-4e95-8edd-f62dd6bc8a6e]
description = "EOW on Monday translates to Friday at five in the afternoon"

[0b375df5-d198-489e-acee-fd538a768616]
description = "EOW on Tuesday translates to Friday at five in the afternoon"

[4afbb881-0b5c-46be-94e1-992cdc2a8ca4]
description = "EOW on Wednesday translates to Friday at five in the afternoon"

[e1341c2b-5e1b-4702-a95c-a01e8e96e510]
description = "EOW on Thursday translates to Sunday at eight in the evening"

[bbffccf7-97f7-4244-888d-bdd64348fa2e]
description = "EOW on Friday translates to Sunday at eight in the evening"

[d651fcf4-290e-407c-8107-36b9076f39b2]
description = "EOW translates to leap day"

[439bf09f-3a0e-44e7-bad5-b7b6d0c4505a]
description = "2M before the second month of this year translates to the first workday of the second month of this year"

[86d82e83-c481-4fb4-9264-625de7521340]
description = "11M in the eleventh month translates to the first workday of the eleventh month of next year"

[0d0b8f6a-1915-46f5-a630-1ff06af9da08]
description = "4M in the ninth month translates to the first workday of the fourth month of next year"

[06d401e3-8461-438f-afae-8d26aa0289e0]
description = "Q1 in the first quarter translates to the last workday of the first quarter of this year"

[eebd5f32-b16d-4ecd-91a0-584b0364b7ed]
description = "Q4 in the second quarter translates to the last workday of the fourth quarter of this year"

[c920886c-44ad-4d34-a156-dc4176186581]
description = "Q3 in the fourth quarter translates to the last workday of the third quarter of next year"
33 changes: 33 additions & 0 deletions exercises/practice/swift-scheduling/SwiftScheduling.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
<?php

/*
* By adding type hints and enabling strict type checking, code can become
* easier to read, self-documenting and reduce the number of potential bugs.
* By default, type declarations are non-strict, which means they will attempt
* to change the original type to match the type specified by the
* type-declaration.
*
* In other words, if you pass a string to a function requiring a float,
* it will attempt to convert the string value to a float.
*
* To enable strict mode, a single declare directive must be placed at the top
* of the file.
* This means that the strictness of typing is configured on a per-file basis.
* This directive not only affects the type declarations of parameters, but also
* a function's return type.
*
* For more info review the Concept on strict type checking in the PHP track
* <link>.
*
* To disable strict typing, comment out the directive below.
*/

declare(strict_types=1);

class SwiftScheduling
{
public function deliveryDate(DateTime $start, string $description): DateTime
{
throw new \BadMethodCallException("Implement the deliveryDate method");
}
}
Loading
Loading