Skip to content

Add swift-scheduling exercise #1551

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

Merged
merged 10 commits into from
Mar 4, 2025
Merged
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
8 changes: 4 additions & 4 deletions .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,7 @@ jobs:
echo "::set-output name=cache_key::$cache_key"

- name: Retrieve Mix Dependencies Cache
uses: actions/cache@6849a6489940f00c2f30c0fb92c6274307ccb58a
uses: actions/cache@d4323d4df104b026a6aa633fdb11d772146be0bf
id: mix-cache # id to use in retrieve action
with:
path: deps
Expand All @@ -55,7 +55,7 @@ jobs:
run: mix

- name: Retrieve PLT Cache
uses: actions/cache@6849a6489940f00c2f30c0fb92c6274307ccb58a
uses: actions/cache@d4323d4df104b026a6aa633fdb11d772146be0bf
id: plt-cache
with:
path: priv/plts
Expand Down Expand Up @@ -106,7 +106,7 @@ jobs:
echo "::set-output name=cache_key::$cache_key"

- name: Retrieve Mix Dependencies Cache
uses: actions/cache@6849a6489940f00c2f30c0fb92c6274307ccb58a
uses: actions/cache@d4323d4df104b026a6aa633fdb11d772146be0bf
id: mix-cache # id to use in retrieve action
with:
path: deps
Expand All @@ -129,7 +129,7 @@ jobs:
- uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683

- name: Retrieve Lychee Cache
uses: actions/cache@6849a6489940f00c2f30c0fb92c6274307ccb58a
uses: actions/cache@d4323d4df104b026a6aa633fdb11d772146be0bf
id: lychee-cache # id to use in retrieve action
with:
path: .lycheecache
Expand Down
17 changes: 17 additions & 0 deletions config.json
Original file line number Diff line number Diff line change
Expand Up @@ -2505,6 +2505,23 @@
],
"difficulty": 6
},
{
"slug": "swift-scheduling",
"name": "Swift Scheduling",
"uuid": "0eb21dfc-9666-475c-bc92-558f2663a598",
"practices": [
"dates-and-time"
],
"prerequisites": [
"strings",
"regular-expressions",
"dates-and-time",
"cond",
"case",
"if"
],
"difficulty": 6
},
{
"slug": "variable-length-quantity",
"name": "Variable Length Quantity",
Expand Down
43 changes: 43 additions & 0 deletions exercises/practice/swift-scheduling/.docs/instructions.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
# 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² |

¹ A workday is a Monday, Tuesday, Wednesday, Thursday, or Friday.
² A year has four quarters, each with three months: January/February/March, April/May/June, July/August/September, and 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.
4 changes: 4 additions & 0 deletions exercises/practice/swift-scheduling/.formatter.exs
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
# Used by "mix format"
[
inputs: ["{mix,.formatter}.exs", "{config,lib,test}/**/*.{ex,exs}"]
]
24 changes: 24 additions & 0 deletions exercises/practice/swift-scheduling/.gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
# The directory Mix will write compiled artifacts to.
/_build/

# If you run "mix test --cover", coverage assets end up here.
/cover/

# The directory Mix downloads your dependencies sources to.
/deps/

# Where third-party dependencies like ExDoc output generated docs.
/doc/

# Ignore .fetch files in case you like to edit your project deps locally.
/.fetch

# If the VM crashes, it generates a dump, let's ignore it too.
erl_crash.dump

# Also ignore archive artifacts (built via "mix archive.build").
*.ez

# Ignore package tarball (built via "mix hex.build").
anagram-*.tar

20 changes: 20 additions & 0 deletions exercises/practice/swift-scheduling/.meta/config.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
{
"authors": [
"angelikatyborska"
],
"contributors": [
"neenjaw"
],
"files": {
"solution": [
"lib/swift_scheduling.ex"
],
"test": [
"test/swift_scheduling_test.exs"
],
"example": [
".meta/example.ex"
]
},
"blurb": "Convert delivery date descriptions to actual delivery dates."
}
134 changes: 134 additions & 0 deletions exercises/practice/swift-scheduling/.meta/example.ex
Original file line number Diff line number Diff line change
@@ -0,0 +1,134 @@
defmodule SwiftScheduling do
@spec delivery_date(NaiveDateTime.t(), String.t()) :: NaiveDateTime.t()
def delivery_date(meeting_date, description) do
cond do
description == "NOW" -> now(meeting_date)
description == "ASAP" -> asap(meeting_date)
description == "EOW" -> eow(meeting_date)
match = Regex.run(~r[^(\d+)M$], description) -> nth_month(meeting_date, match)
match = Regex.run(~r[^Q(\d+)$], description) -> nth_quarter(meeting_date, match)
end
end

defp now(meeting_date) do
NaiveDateTime.add(meeting_date, 2, :hour)
end

defp asap(meeting_date) do
meet_today? =
meeting_date
|> NaiveDateTime.to_time()
|> Time.compare(~T[13:00:00])
|> Kernel.==(:lt)

[new_date, new_time] =
if meet_today? do
[meeting_date |> NaiveDateTime.to_date(), ~T[17:00:00]]
else
[
meeting_date
|> NaiveDateTime.to_date()
|> Date.add(1),
~T[13:00:00]
]
end

NaiveDateTime.new!(new_date, new_time)
end

defp eow(meeting_date) do
before_thursday? =
meeting_date
|> NaiveDateTime.to_date()
|> Date.day_of_week()
|> Kernel.<(4)

sunday =
meeting_date
|> NaiveDateTime.to_date()
|> Date.end_of_week()

[new_date, new_time] =
if before_thursday? do
friday = Date.add(sunday, -2)
[friday, ~T[17:00:00]]
else
[sunday, ~T[20:00:00]]
end

NaiveDateTime.new!(new_date, new_time)
end

defp nth_month(meeting_date, [_, month]) do
month_number = String.to_integer(month)
schedule_for_this_year? = meeting_date.month < month_number

new_time = ~T[08:00:00]

nth_month_this_year = %{NaiveDateTime.to_date(meeting_date) | month: month_number}

new_date =
if schedule_for_this_year? do
first_workday_of_month(nth_month_this_year)
else
nth_month_next_year = nth_month_this_year |> shift_year(1)
first_workday_of_month(nth_month_next_year)
end

NaiveDateTime.new!(new_date, new_time)
end

defp nth_quarter(meeting_date, [_, quarter]) do
quarter_number = String.to_integer(quarter)
schedule_for_this_year? = meeting_date.month <= quarter_number * 3

new_time = ~T[08:00:00]

last_month_of_nth_quarter_this_year = %{
NaiveDateTime.to_date(meeting_date)
| month: quarter_number * 3
}

new_date =
if schedule_for_this_year? do
last_workday_of_month(last_month_of_nth_quarter_this_year)
else
last_month_of_nth_quarter_next_year =
last_month_of_nth_quarter_this_year
|> shift_year(1)

last_workday_of_month(last_month_of_nth_quarter_next_year)
end

NaiveDateTime.new!(new_date, new_time)
end

defp first_workday_of_month(date) do
first_day_of_month =
date
|> Date.beginning_of_month()

case Date.day_of_week(first_day_of_month) do
n when n in 1..5 -> first_day_of_month
6 -> Date.add(first_day_of_month, 2)
7 -> Date.add(first_day_of_month, 1)
end
end

defp last_workday_of_month(date) do
first_day_of_month =
date
|> Date.end_of_month()

case Date.day_of_week(first_day_of_month) do
n when n in 1..5 -> first_day_of_month
6 -> Date.add(first_day_of_month, -1)
7 -> Date.add(first_day_of_month, -2)
end
end

# Date.shift is only available from 1.17
defp shift_year(date, n) do
%{date | year: date.year + n}
end
end
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"
8 changes: 8 additions & 0 deletions exercises/practice/swift-scheduling/lib/swift_scheduling.ex
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
defmodule SwiftScheduling do
@doc """
Convert delivery date descriptions to actual delivery dates, based on when the meeting started.
"""
@spec delivery_date(NaiveDateTime.t(), String.t()) :: NaiveDateTime.t()
def delivery_date(meeting_date, description) do
end
end
28 changes: 28 additions & 0 deletions exercises/practice/swift-scheduling/mix.exs
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
defmodule SwiftScheduling.MixProject do
use Mix.Project

def project do
[
app: :swift_scheduling,
version: "0.1.0",
# elixir: "~> 1.8",
start_permanent: Mix.env() == :prod,
deps: deps()
]
end

# Run "mix help compile.app" to learn about applications.
def application do
[
extra_applications: [:logger]
]
end

# Run "mix help deps" to learn about dependencies.
defp deps do
[
# {:dep_from_hexpm, "~> 0.3.0"},
# {:dep_from_git, git: "https://github.com/elixir-lang/my_dep.git", tag: "0.1.0"}
]
end
end
Loading
Loading