Skip to content

Commit 291ff76

Browse files
committed
Add split-second-stopwatch
1 parent 7f21eb9 commit 291ff76

10 files changed

Lines changed: 555 additions & 0 deletions

File tree

config.json

Lines changed: 13 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1489,6 +1489,19 @@
14891489
"practices": [],
14901490
"prerequisites": [],
14911491
"difficulty": 3
1492+
},
1493+
{
1494+
"slug": "split-second-stopwatch",
1495+
"name": "Split-Second Stopwatch",
1496+
"uuid": "49a0daf5-ecf8-41ad-8517-f969b770bf32",
1497+
"practices": [],
1498+
"prerequisites": [],
1499+
"difficulty": 3,
1500+
"topics": [
1501+
"strings",
1502+
"text-formatting",
1503+
"time"
1504+
]
14921505
}
14931506
]
14941507
},
Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
return {
2+
default = {
3+
ROOT = { '.' }
4+
}
5+
}
Lines changed: 22 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
# Instructions
2+
3+
Your task is to build a stopwatch to keep precise track of lap times.
4+
5+
The stopwatch uses four commands (start, stop, lap, and reset) to keep track of:
6+
7+
1. The current lap's tracked time
8+
2. Previously recorded lap times
9+
10+
What commands can be used depends on which state the stopwatch is in:
11+
12+
1. Ready: initial state
13+
2. Running: tracking time
14+
3. Stopped: not tracking time
15+
16+
| Command | Begin state | End state | Effect |
17+
| ------- | ----------- | --------- | -------------------------------------------------------- |
18+
| Start | Ready | Running | Start tracking time |
19+
| Start | Stopped | Running | Resume tracking time |
20+
| Stop | Running | Stopped | Stop tracking time |
21+
| Lap | Running | Running | Add current lap to previous laps, then reset current lap |
22+
| Reset | Stopped | Ready | Reset current lap and clear previous laps |
Lines changed: 6 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
# Introduction
2+
3+
You've always run for the thrill of it — no schedules, no timers, just the sound of your feet on the pavement.
4+
But now that you've joined a competitive running crew, things are getting serious.
5+
Training sessions are timed to the second, and every split second counts.
6+
To keep pace, you've picked up the _Split-Second Stopwatch_ — a sleek, high-tech gadget that's about to become your new best friend.
Lines changed: 19 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
{
2+
"authors": [
3+
"ryanplusplus"
4+
],
5+
"files": {
6+
"solution": [
7+
"split-second-stopwatch.lua"
8+
],
9+
"test": [
10+
"split-second-stopwatch_spec.lua"
11+
],
12+
"example": [
13+
".meta/example.lua"
14+
]
15+
},
16+
"blurb": "Keep track of time through a digital stopwatch.",
17+
"source": "Erik Schierboom",
18+
"source_url": "https://github.com/exercism/problem-specifications/pull/2547"
19+
}
Lines changed: 71 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,71 @@
1+
local function time_from_seconds(seconds)
2+
local hours = math.floor(seconds / 3600)
3+
local minutes = math.floor((seconds % 3600) / 60)
4+
local secs = seconds % 60
5+
return string.format('%02d:%02d:%02d', hours, minutes, secs)
6+
end
7+
8+
local function seconds_from_time(timestamp)
9+
local hours, minutes, seconds = timestamp:match("(%d+):(%d+):(%d+)")
10+
return tonumber(hours) * 3600 + tonumber(minutes) * 60 + tonumber(seconds)
11+
end
12+
13+
local Stopwatch = {}
14+
15+
function Stopwatch:new()
16+
local obj = { _state = 'stopped' }
17+
setmetatable(obj, self)
18+
self.__index = self
19+
obj:reset()
20+
return obj
21+
end
22+
23+
function Stopwatch:start()
24+
assert(self._state ~= 'running', 'cannot start an already running stopwatch')
25+
self._state = 'running'
26+
end
27+
28+
function Stopwatch:stop()
29+
assert(self._state == 'running', 'cannot stop a stopwatch that is not running')
30+
self._state = 'stopped'
31+
end
32+
33+
function Stopwatch:reset()
34+
assert(self._state == 'stopped', 'cannot reset a stopwatch that is not stopped')
35+
self._state = 'ready'
36+
self._total_time = 0
37+
self._lap_time = 0
38+
self._previous_laps = {}
39+
end
40+
41+
function Stopwatch:advance_time(timestamp)
42+
if self._state == 'running' then
43+
local time_in_seconds = seconds_from_time(timestamp)
44+
self._total_time = self._total_time + time_in_seconds
45+
self._lap_time = self._lap_time + time_in_seconds
46+
end
47+
end
48+
49+
function Stopwatch:total()
50+
return time_from_seconds(self._total_time)
51+
end
52+
53+
function Stopwatch:lap()
54+
assert(self._state == 'running', 'cannot lap a stopwatch that is not running')
55+
table.insert(self._previous_laps, time_from_seconds(self._lap_time))
56+
self._lap_time = 0
57+
end
58+
59+
function Stopwatch:current_lap()
60+
return time_from_seconds(self._lap_time)
61+
end
62+
63+
function Stopwatch:previous_laps()
64+
return self._previous_laps
65+
end
66+
67+
function Stopwatch:state()
68+
return self._state
69+
end
70+
71+
return Stopwatch
Lines changed: 37 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
local utils = require 'utils'
2+
3+
local function render_expected(expected)
4+
if type(expected) == 'table' then
5+
return '{ ' .. table.concat(utils.map(expected, utils.stringify), ', ') .. '}'
6+
else
7+
return utils.stringify(expected)
8+
end
9+
end
10+
11+
return {
12+
module_name = 'Stopwatch',
13+
14+
generate_test = function(case)
15+
local lines = {}
16+
17+
for _, command in ipairs(case.input.commands) do
18+
if command.command == 'new' then
19+
table.insert(lines, 'local stopwatch = Stopwatch:new()')
20+
elseif command.expected and command.expected.error then
21+
table.insert(lines,
22+
('assert.has_error(function() stopwatch:%s() end, %s)'):format(utils.snake_case(command.command),
23+
utils.stringify(
24+
command.expected.error)))
25+
elseif command.by then
26+
table.insert(lines, ('stopwatch:%s(%s)'):format(utils.snake_case(command.command), render_expected(command.by)))
27+
elseif command.expected then
28+
table.insert(lines, ('assert.are.same(%s, stopwatch:%s())'):format(render_expected(command.expected),
29+
utils.snake_case(command.command)))
30+
else
31+
table.insert(lines, ('stopwatch:%s()'):format(utils.snake_case(command.command)))
32+
end
33+
end
34+
35+
return table.concat(lines, '\n')
36+
end
37+
}
Lines changed: 97 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,97 @@
1+
# This is an auto-generated file.
2+
#
3+
# Regenerating this file via `configlet sync` will:
4+
# - Recreate every `description` key/value pair
5+
# - Recreate every `reimplements` key/value pair, where they exist in problem-specifications
6+
# - Remove any `include = true` key/value pair (an omitted `include` key implies inclusion)
7+
# - Preserve any other key/value pair
8+
#
9+
# As user-added comments (using the # character) will be removed when this file
10+
# is regenerated, comments can be added via a `comment` key.
11+
12+
[ddb238ea-99d4-4eaa-a81d-3c917a525a23]
13+
description = "new stopwatch starts in ready state"
14+
15+
[b19635d4-08ad-4ac3-b87f-aca10e844071]
16+
description = "new stopwatch's current lap has no elapsed time"
17+
18+
[492eb532-268d-43ea-8a19-2a032067d335]
19+
description = "new stopwatch's total has no elapsed time"
20+
21+
[8a892c1e-9ef7-4690-894e-e155a1fe4484]
22+
description = "new stopwatch does not have previous laps"
23+
24+
[5b2705b6-a584-4042-ba3a-4ab8d0ab0281]
25+
description = "start from ready state changes state to running"
26+
27+
[748235ce-1109-440b-9898-0a431ea179b6]
28+
description = "start does not change previous laps"
29+
30+
[491487b1-593d-423e-a075-aa78d449ff1f]
31+
description = "start initiates time tracking for current lap"
32+
33+
[a0a7ba2c-8db6-412c-b1b6-cb890e9b72ed]
34+
description = "start initiates time tracking for total"
35+
36+
[7f558a17-ef6d-4a5b-803a-f313af7c41d3]
37+
description = "start cannot be called from running state"
38+
39+
[32466eef-b2be-4d60-a927-e24fce52dab9]
40+
description = "stop from running state changes state to stopped"
41+
42+
[621eac4c-8f43-4d99-919c-4cad776d93df]
43+
description = "stop pauses time tracking for current lap"
44+
45+
[465bcc82-7643-41f2-97ff-5e817cef8db4]
46+
description = "stop pauses time tracking for total"
47+
48+
[b1ba7454-d627-41ee-a078-891b2ed266fc]
49+
description = "stop cannot be called from ready state"
50+
51+
[5c041078-0898-44dc-9d5b-8ebb5352626c]
52+
description = "stop cannot be called from stopped state"
53+
54+
[3f32171d-8fbf-46b6-bc2b-0810e1ec53b7]
55+
description = "start from stopped state changes state to running"
56+
57+
[626997cb-78d5-4fe8-b501-29fdef804799]
58+
description = "start from stopped state resumes time tracking for current lap"
59+
60+
[58487c53-ab26-471c-a171-807ef6363319]
61+
description = "start from stopped state resumes time tracking for total"
62+
63+
[091966e3-ed25-4397-908b-8bb0330118f8]
64+
description = "lap adds current lap to previous laps"
65+
66+
[1aa4c5ee-a7d5-4d59-9679-419deef3c88f]
67+
description = "lap resets current lap and resumes time tracking"
68+
69+
[4b46b92e-1b3f-46f6-97d2-0082caf56e80]
70+
description = "lap continues time tracking for total"
71+
72+
[ea75d36e-63eb-4f34-97ce-8c70e620bdba]
73+
description = "lap cannot be called from ready state"
74+
75+
[63731154-a23a-412d-a13f-c562f208eb1e]
76+
description = "lap cannot be called from stopped state"
77+
78+
[e585ee15-3b3f-4785-976b-dd96e7cc978b]
79+
description = "stop does not change previous laps"
80+
81+
[fc3645e2-86cf-4d11-97c6-489f031103f6]
82+
description = "reset from stopped state changes state to ready"
83+
84+
[20fbfbf7-68ad-4310-975a-f5f132886c4e]
85+
description = "reset resets current lap"
86+
87+
[00a8f7bb-dd5c-43e5-8705-3ef124007662]
88+
description = "reset clears previous laps"
89+
90+
[76cea936-6214-4e95-b6d1-4d4edcf90499]
91+
description = "reset cannot be called from ready state"
92+
93+
[ba4d8e69-f200-4721-b59e-90d8cf615153]
94+
description = "reset cannot be called from running state"
95+
96+
[0b01751a-cb57-493f-bb86-409de6e84306]
97+
description = "supports very long laps"
Lines changed: 37 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
local Stopwatch = {}
2+
3+
function Stopwatch:new()
4+
local obj = {}
5+
setmetatable(obj, self)
6+
self.__index = self
7+
return obj
8+
end
9+
10+
function Stopwatch:start()
11+
end
12+
13+
function Stopwatch:stop()
14+
end
15+
16+
function Stopwatch:reset()
17+
end
18+
19+
function Stopwatch:advance_time(timestamp)
20+
end
21+
22+
function Stopwatch:total()
23+
end
24+
25+
function Stopwatch:lap()
26+
end
27+
28+
function Stopwatch:current_lap()
29+
end
30+
31+
function Stopwatch:previous_laps()
32+
end
33+
34+
function Stopwatch:state()
35+
end
36+
37+
return Stopwatch

0 commit comments

Comments
 (0)